commit bc0c412f30e18f2069e1b9081f2f7728261c12ed Author: Bojan Mojsilovic Date: Fri Jul 7 16:14:22 2023 +0200 Opening my eyes diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..76add87 --- /dev/null +++ b/.gitignore @@ -0,0 +1,2 @@ +node_modules +dist \ No newline at end of file diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..a14b6f6 --- /dev/null +++ b/LICENSE @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) 2023 PRIMAL SYSTEMS INC. + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/README.md b/README.md new file mode 100644 index 0000000..2643c2a --- /dev/null +++ b/README.md @@ -0,0 +1,16 @@ +# Primal Web Client + +This repo holds the web client of the Primal Nostr. + +## Setup + +- Clone this repo +- run `npm install` +- run `npm run dev` to stand a local instance + +## Development + +This code is still very much a work-in-progress. Expect major changes of structure and logic to be happening somewhat frequently. +Major features are still missing. + +This code is provided as-is. diff --git a/fonts-google.css b/fonts-google.css new file mode 100644 index 0000000..823d09a --- /dev/null +++ b/fonts-google.css @@ -0,0 +1,181 @@ +/* cyrillic-ext */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 100; + font-stretch: 100%; + font-display: swap; + src: url(https://fonts.gstatic.com/s/robotoflex/v9/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-wmF9lp.woff2) format('woff2'); + unicode-range: U+0460-052F, U+1C80-1C88, U+20B4, U+2DE0-2DFF, U+A640-A69F, U+FE2E-FE2F; +} +/* cyrillic */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 100; + font-stretch: 100%; + font-display: swap; + src: url(https://fonts.gstatic.com/s/robotoflex/v9/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-UmF9lp.woff2) format('woff2'); + unicode-range: U+0301, U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116; +} +/* greek */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 100; + font-stretch: 100%; + font-display: swap; + src: url(https://fonts.gstatic.com/s/robotoflex/v9/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-ImF9lp.woff2) format('woff2'); + unicode-range: U+0370-03FF; +} +/* vietnamese */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 100; + font-stretch: 100%; + font-display: swap; + src: url(https://fonts.gstatic.com/s/robotoflex/v9/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-4mF9lp.woff2) format('woff2'); + unicode-range: U+0102-0103, U+0110-0111, U+0128-0129, U+0168-0169, U+01A0-01A1, U+01AF-01B0, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329, U+1EA0-1EF9, U+20AB; +} +/* latin-ext */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 100; + font-stretch: 100%; + font-display: swap; + src: url(https://fonts.gstatic.com/s/robotoflex/v9/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-8mF9lp.woff2) format('woff2'); + unicode-range: U+0100-02AF, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329, U+1E00-1EFF, U+2020, U+20A0-20AB, U+20AD-20CF, U+2113, U+2C60-2C7F, U+A720-A7FF; +} +/* latin */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 100; + font-stretch: 100%; + font-display: swap; + src: url(https://fonts.gstatic.com/s/robotoflex/v9/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-EmFw.woff2) format('woff2'); + unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD; +} +/* cyrillic-ext */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 400; + font-stretch: 100%; + font-display: swap; + src: url(https://fonts.gstatic.com/s/robotoflex/v9/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-wmF9lp.woff2) format('woff2'); + unicode-range: U+0460-052F, U+1C80-1C88, U+20B4, U+2DE0-2DFF, U+A640-A69F, U+FE2E-FE2F; +} +/* cyrillic */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 400; + font-stretch: 100%; + font-display: swap; + src: url(https://fonts.gstatic.com/s/robotoflex/v9/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-UmF9lp.woff2) format('woff2'); + unicode-range: U+0301, U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116; +} +/* greek */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 400; + font-stretch: 100%; + font-display: swap; + src: url(https://fonts.gstatic.com/s/robotoflex/v9/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-ImF9lp.woff2) format('woff2'); + unicode-range: U+0370-03FF; +} +/* vietnamese */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 400; + font-stretch: 100%; + font-display: swap; + src: url(https://fonts.gstatic.com/s/robotoflex/v9/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-4mF9lp.woff2) format('woff2'); + unicode-range: U+0102-0103, U+0110-0111, U+0128-0129, U+0168-0169, U+01A0-01A1, U+01AF-01B0, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329, U+1EA0-1EF9, U+20AB; +} +/* latin-ext */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 400; + font-stretch: 100%; + font-display: swap; + src: url(https://fonts.gstatic.com/s/robotoflex/v9/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-8mF9lp.woff2) format('woff2'); + unicode-range: U+0100-02AF, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329, U+1E00-1EFF, U+2020, U+20A0-20AB, U+20AD-20CF, U+2113, U+2C60-2C7F, U+A720-A7FF; +} +/* latin */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 400; + font-stretch: 100%; + font-display: swap; + src: url(https://fonts.gstatic.com/s/robotoflex/v9/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-EmFw.woff2) format('woff2'); + unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD; +} +/* cyrillic-ext */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 700; + font-stretch: 100%; + font-display: swap; + src: url(https://fonts.gstatic.com/s/robotoflex/v9/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-wmF9lp.woff2) format('woff2'); + unicode-range: U+0460-052F, U+1C80-1C88, U+20B4, U+2DE0-2DFF, U+A640-A69F, U+FE2E-FE2F; +} +/* cyrillic */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 700; + font-stretch: 100%; + font-display: swap; + src: url(https://fonts.gstatic.com/s/robotoflex/v9/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-UmF9lp.woff2) format('woff2'); + unicode-range: U+0301, U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116; +} +/* greek */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 700; + font-stretch: 100%; + font-display: swap; + src: url(https://fonts.gstatic.com/s/robotoflex/v9/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-ImF9lp.woff2) format('woff2'); + unicode-range: U+0370-03FF; +} +/* vietnamese */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 700; + font-stretch: 100%; + font-display: swap; + src: url(https://fonts.gstatic.com/s/robotoflex/v9/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-4mF9lp.woff2) format('woff2'); + unicode-range: U+0102-0103, U+0110-0111, U+0128-0129, U+0168-0169, U+01A0-01A1, U+01AF-01B0, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329, U+1EA0-1EF9, U+20AB; +} +/* latin-ext */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 700; + font-stretch: 100%; + font-display: swap; + src: url(https://fonts.gstatic.com/s/robotoflex/v9/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-8mF9lp.woff2) format('woff2'); + unicode-range: U+0100-02AF, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329, U+1E00-1EFF, U+2020, U+20A0-20AB, U+20AD-20CF, U+2113, U+2C60-2C7F, U+A720-A7FF; +} +/* latin */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 700; + font-stretch: 100%; + font-display: swap; + src: url(https://fonts.gstatic.com/s/robotoflex/v9/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-EmFw.woff2) format('woff2'); + unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD; +} + diff --git a/index.html b/index.html new file mode 100644 index 0000000..4bcad56 --- /dev/null +++ b/index.html @@ -0,0 +1,24 @@ + + + + + + + + + + + + + + + Primal + + + +
+ + + + + diff --git a/package-lock.json b/package-lock.json new file mode 100644 index 0000000..d112be7 --- /dev/null +++ b/package-lock.json @@ -0,0 +1,2298 @@ +{ + "name": "vite-template-solid", + "version": "0.0.0", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "name": "vite-template-solid", + "version": "0.0.0", + "license": "MIT", + "dependencies": { + "@cookbook/solid-intl": "^0.1.2", + "@jukben/emoji-search": "^2.0.1", + "@picocss/pico": "^1.5.7", + "@scure/base": "^1.1.1", + "@solidjs/router": "^0.7.0", + "@thisbeyond/solid-select": "^0.13.0", + "@types/dompurify": "^2.4.0", + "dompurify": "^3.0.0", + "link-preview-js": "^3.0.4", + "nostr-tools": "^1.4.1", + "sass": "^1.58.0", + "solid-js": "^1.6.6" + }, + "devDependencies": { + "@formatjs/cli": "^6.0.4", + "typescript": "^4.9.4", + "vite": "^4.0.3", + "vite-plugin-solid": "^2.5.0" + } + }, + "node_modules/@ampproject/remapping": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/@ampproject/remapping/-/remapping-2.2.0.tgz", + "integrity": "sha512-qRmjj8nj9qmLTQXXmaR1cck3UXSRMPrbsLJAasZpF+t3riI71BXed5ebIOYwQntykeZuhjsdweEc9BxH5Jc26w==", + "dev": true, + "dependencies": { + "@jridgewell/gen-mapping": "^0.1.0", + "@jridgewell/trace-mapping": "^0.3.9" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@babel/code-frame": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.18.6.tgz", + "integrity": "sha512-TDCmlK5eOvH+eH7cdAFlNXeVJqWIQ7gW9tY1GJIpUtFb6CmjVyq2VM3u71bOyR8CRihcCgMUYoDNyLXao3+70Q==", + "dev": true, + "dependencies": { + "@babel/highlight": "^7.18.6" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/compat-data": { + "version": "7.20.14", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.20.14.tgz", + "integrity": "sha512-0YpKHD6ImkWMEINCyDAD0HLLUH/lPCefG8ld9it8DJB2wnApraKuhgYTvTY1z7UFIfBTGy5LwncZ+5HWWGbhFw==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/core": { + "version": "7.20.12", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.20.12.tgz", + "integrity": "sha512-XsMfHovsUYHFMdrIHkZphTN/2Hzzi78R08NuHfDBehym2VsPDL6Zn/JAD/JQdnRvbSsbQc4mVaU1m6JgtTEElg==", + "dev": true, + "dependencies": { + "@ampproject/remapping": "^2.1.0", + "@babel/code-frame": "^7.18.6", + "@babel/generator": "^7.20.7", + "@babel/helper-compilation-targets": "^7.20.7", + "@babel/helper-module-transforms": "^7.20.11", + "@babel/helpers": "^7.20.7", + "@babel/parser": "^7.20.7", + "@babel/template": "^7.20.7", + "@babel/traverse": "^7.20.12", + "@babel/types": "^7.20.7", + "convert-source-map": "^1.7.0", + "debug": "^4.1.0", + "gensync": "^1.0.0-beta.2", + "json5": "^2.2.2", + "semver": "^6.3.0" + }, + "engines": { + "node": ">=6.9.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/babel" + } + }, + "node_modules/@babel/generator": { + "version": "7.20.14", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.20.14.tgz", + "integrity": "sha512-AEmuXHdcD3A52HHXxaTmYlb8q/xMEhoRP67B3T4Oq7lbmSoqroMZzjnGj3+i1io3pdnF8iBYVu4Ilj+c4hBxYg==", + "dev": true, + "dependencies": { + "@babel/types": "^7.20.7", + "@jridgewell/gen-mapping": "^0.3.2", + "jsesc": "^2.5.1" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/generator/node_modules/@jridgewell/gen-mapping": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.2.tgz", + "integrity": "sha512-mh65xKQAzI6iBcFzwv28KVWSmCkdRBWoOh+bYQGW3+6OZvbbN3TqMGo5hqYxQniRcH9F2VZIoJCm4pa3BPDK/A==", + "dev": true, + "dependencies": { + "@jridgewell/set-array": "^1.0.1", + "@jridgewell/sourcemap-codec": "^1.4.10", + "@jridgewell/trace-mapping": "^0.3.9" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@babel/helper-annotate-as-pure": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/helper-annotate-as-pure/-/helper-annotate-as-pure-7.18.6.tgz", + "integrity": "sha512-duORpUiYrEpzKIop6iNbjnwKLAKnJ47csTyRACyEmWj0QdUrm5aqNJGHSSEQSUAvNW0ojX0dOmK9dZduvkfeXA==", + "dev": true, + "dependencies": { + "@babel/types": "^7.18.6" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-compilation-targets": { + "version": "7.20.7", + "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.20.7.tgz", + "integrity": "sha512-4tGORmfQcrc+bvrjb5y3dG9Mx1IOZjsHqQVUz7XCNHO+iTmqxWnVg3KRygjGmpRLJGdQSKuvFinbIb0CnZwHAQ==", + "dev": true, + "dependencies": { + "@babel/compat-data": "^7.20.5", + "@babel/helper-validator-option": "^7.18.6", + "browserslist": "^4.21.3", + "lru-cache": "^5.1.1", + "semver": "^6.3.0" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-create-class-features-plugin": { + "version": "7.20.12", + "resolved": "https://registry.npmjs.org/@babel/helper-create-class-features-plugin/-/helper-create-class-features-plugin-7.20.12.tgz", + "integrity": "sha512-9OunRkbT0JQcednL0UFvbfXpAsUXiGjUk0a7sN8fUXX7Mue79cUSMjHGDRRi/Vz9vYlpIhLV5fMD5dKoMhhsNQ==", + "dev": true, + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.18.6", + "@babel/helper-environment-visitor": "^7.18.9", + "@babel/helper-function-name": "^7.19.0", + "@babel/helper-member-expression-to-functions": "^7.20.7", + "@babel/helper-optimise-call-expression": "^7.18.6", + "@babel/helper-replace-supers": "^7.20.7", + "@babel/helper-skip-transparent-expression-wrappers": "^7.20.0", + "@babel/helper-split-export-declaration": "^7.18.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-environment-visitor": { + "version": "7.18.9", + "resolved": "https://registry.npmjs.org/@babel/helper-environment-visitor/-/helper-environment-visitor-7.18.9.tgz", + "integrity": "sha512-3r/aACDJ3fhQ/EVgFy0hpj8oHyHpQc+LPtJoY9SzTThAsStm4Ptegq92vqKoE3vD706ZVFWITnMnxucw+S9Ipg==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-function-name": { + "version": "7.19.0", + "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.19.0.tgz", + "integrity": "sha512-WAwHBINyrpqywkUH0nTnNgI5ina5TFn85HKS0pbPDfxFfhyR/aNQEn4hGi1P1JyT//I0t4OgXUlofzWILRvS5w==", + "dev": true, + "dependencies": { + "@babel/template": "^7.18.10", + "@babel/types": "^7.19.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-hoist-variables": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/helper-hoist-variables/-/helper-hoist-variables-7.18.6.tgz", + "integrity": "sha512-UlJQPkFqFULIcyW5sbzgbkxn2FKRgwWiRexcuaR8RNJRy8+LLveqPjwZV/bwrLZCN0eUHD/x8D0heK1ozuoo6Q==", + "dev": true, + "dependencies": { + "@babel/types": "^7.18.6" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-member-expression-to-functions": { + "version": "7.20.7", + "resolved": "https://registry.npmjs.org/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.20.7.tgz", + "integrity": "sha512-9J0CxJLq315fEdi4s7xK5TQaNYjZw+nDVpVqr1axNGKzdrdwYBD5b4uKv3n75aABG0rCCTK8Im8Ww7eYfMrZgw==", + "dev": true, + "dependencies": { + "@babel/types": "^7.20.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-imports": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.18.6.tgz", + "integrity": "sha512-0NFvs3VkuSYbFi1x2Vd6tKrywq+z/cLeYC/RJNFrIX/30Bf5aiGYbtvGXolEktzJH8o5E5KJ3tT+nkxuuZFVlA==", + "dev": true, + "dependencies": { + "@babel/types": "^7.18.6" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-transforms": { + "version": "7.20.11", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.20.11.tgz", + "integrity": "sha512-uRy78kN4psmji1s2QtbtcCSaj/LILFDp0f/ymhpQH5QY3nljUZCaNWz9X1dEj/8MBdBEFECs7yRhKn8i7NjZgg==", + "dev": true, + "dependencies": { + "@babel/helper-environment-visitor": "^7.18.9", + "@babel/helper-module-imports": "^7.18.6", + "@babel/helper-simple-access": "^7.20.2", + "@babel/helper-split-export-declaration": "^7.18.6", + "@babel/helper-validator-identifier": "^7.19.1", + "@babel/template": "^7.20.7", + "@babel/traverse": "^7.20.10", + "@babel/types": "^7.20.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-optimise-call-expression": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/helper-optimise-call-expression/-/helper-optimise-call-expression-7.18.6.tgz", + "integrity": "sha512-HP59oD9/fEHQkdcbgFCnbmgH5vIQTJbxh2yf+CdM89/glUNnuzr87Q8GIjGEnOktTROemO0Pe0iPAYbqZuOUiA==", + "dev": true, + "dependencies": { + "@babel/types": "^7.18.6" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-plugin-utils": { + "version": "7.20.2", + "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.20.2.tgz", + "integrity": "sha512-8RvlJG2mj4huQ4pZ+rU9lqKi9ZKiRmuvGuM2HlWmkmgOhbs6zEAw6IEiJ5cQqGbDzGZOhwuOQNtZMi/ENLjZoQ==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-replace-supers": { + "version": "7.20.7", + "resolved": "https://registry.npmjs.org/@babel/helper-replace-supers/-/helper-replace-supers-7.20.7.tgz", + "integrity": "sha512-vujDMtB6LVfNW13jhlCrp48QNslK6JXi7lQG736HVbHz/mbf4Dc7tIRh1Xf5C0rF7BP8iiSxGMCmY6Ci1ven3A==", + "dev": true, + "dependencies": { + "@babel/helper-environment-visitor": "^7.18.9", + "@babel/helper-member-expression-to-functions": "^7.20.7", + "@babel/helper-optimise-call-expression": "^7.18.6", + "@babel/template": "^7.20.7", + "@babel/traverse": "^7.20.7", + "@babel/types": "^7.20.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-simple-access": { + "version": "7.20.2", + "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.20.2.tgz", + "integrity": "sha512-+0woI/WPq59IrqDYbVGfshjT5Dmk/nnbdpcF8SnMhhXObpTq2KNBdLFRFrkVdbDOyUmHBCxzm5FHV1rACIkIbA==", + "dev": true, + "dependencies": { + "@babel/types": "^7.20.2" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-skip-transparent-expression-wrappers": { + "version": "7.20.0", + "resolved": "https://registry.npmjs.org/@babel/helper-skip-transparent-expression-wrappers/-/helper-skip-transparent-expression-wrappers-7.20.0.tgz", + "integrity": "sha512-5y1JYeNKfvnT8sZcK9DVRtpTbGiomYIHviSP3OQWmDPU3DeH4a1ZlT/N2lyQ5P8egjcRaT/Y9aNqUxK0WsnIIg==", + "dev": true, + "dependencies": { + "@babel/types": "^7.20.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-split-export-declaration": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.18.6.tgz", + "integrity": "sha512-bde1etTx6ZyTmobl9LLMMQsaizFVZrquTEHOqKeQESMKo4PlObf+8+JA25ZsIpZhT/WEd39+vOdLXAFG/nELpA==", + "dev": true, + "dependencies": { + "@babel/types": "^7.18.6" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-string-parser": { + "version": "7.19.4", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.19.4.tgz", + "integrity": "sha512-nHtDoQcuqFmwYNYPz3Rah5ph2p8PFeFCsZk9A/48dPc/rGocJ5J3hAAZ7pb76VWX3fZKu+uEr/FhH5jLx7umrw==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-identifier": { + "version": "7.19.1", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.19.1.tgz", + "integrity": "sha512-awrNfaMtnHUr653GgGEs++LlAvW6w+DcPrOliSMXWCKo597CwL5Acf/wWdNkf/tfEQE3mjkeD1YOVZOUV/od1w==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-option": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.18.6.tgz", + "integrity": "sha512-XO7gESt5ouv/LRJdrVjkShckw6STTaB7l9BrpBaAHDeF5YZT+01PCwmR0SJHnkW6i8OwW/EVWRShfi4j2x+KQw==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helpers": { + "version": "7.20.13", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.20.13.tgz", + "integrity": "sha512-nzJ0DWCL3gB5RCXbUO3KIMMsBY2Eqbx8mBpKGE/02PgyRQFcPQLbkQ1vyy596mZLaP+dAfD+R4ckASzNVmW3jg==", + "dev": true, + "dependencies": { + "@babel/template": "^7.20.7", + "@babel/traverse": "^7.20.13", + "@babel/types": "^7.20.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/highlight": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.18.6.tgz", + "integrity": "sha512-u7stbOuYjaPezCuLj29hNW1v64M2Md2qupEKP1fHc7WdOA3DgLh37suiSrZYY7haUB7iBeQZ9P1uiRF359do3g==", + "dev": true, + "dependencies": { + "@babel/helper-validator-identifier": "^7.18.6", + "chalk": "^2.0.0", + "js-tokens": "^4.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/parser": { + "version": "7.20.13", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.20.13.tgz", + "integrity": "sha512-gFDLKMfpiXCsjt4za2JA9oTMn70CeseCehb11kRZgvd7+F67Hih3OHOK24cRrWECJ/ljfPGac6ygXAs/C8kIvw==", + "dev": true, + "bin": { + "parser": "bin/babel-parser.js" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@babel/plugin-syntax-jsx": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-jsx/-/plugin-syntax-jsx-7.18.6.tgz", + "integrity": "sha512-6mmljtAedFGTWu2p/8WIORGwy+61PLgOMPOdazc7YoJ9ZCWUyFy3A6CpPkRKLKD1ToAesxX8KGEViAiLo9N+7Q==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.18.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-typescript": { + "version": "7.20.0", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-typescript/-/plugin-syntax-typescript-7.20.0.tgz", + "integrity": "sha512-rd9TkG+u1CExzS4SM1BlMEhMXwFLKVjOAFFCDx9PbX5ycJWDoWMcwdJH9RhkPu1dOgn5TrxLot/Gx6lWFuAUNQ==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.19.0" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-typescript": { + "version": "7.20.13", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-typescript/-/plugin-transform-typescript-7.20.13.tgz", + "integrity": "sha512-O7I/THxarGcDZxkgWKMUrk7NK1/WbHAg3Xx86gqS6x9MTrNL6AwIluuZ96ms4xeDe6AVx6rjHbWHP7x26EPQBA==", + "dev": true, + "dependencies": { + "@babel/helper-create-class-features-plugin": "^7.20.12", + "@babel/helper-plugin-utils": "^7.20.2", + "@babel/plugin-syntax-typescript": "^7.20.0" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/preset-typescript": { + "version": "7.18.6", + "resolved": "https://registry.npmjs.org/@babel/preset-typescript/-/preset-typescript-7.18.6.tgz", + "integrity": "sha512-s9ik86kXBAnD760aybBucdpnLsAt0jK1xqJn2juOn9lkOvSHV60os5hxoVJsPzMQxvnUJFAlkont2DvvaYEBtQ==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.18.6", + "@babel/helper-validator-option": "^7.18.6", + "@babel/plugin-transform-typescript": "^7.18.6" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/runtime": { + "version": "7.5.5", + "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.5.5.tgz", + "integrity": "sha512-28QvEGyQyNkB0/m2B4FU7IEZGK2NUrcMtT6BZEFALTguLk+AUT6ofsHtPk5QyjAdUkpMJ+/Em+quwz4HOt30AQ==", + "dependencies": { + "regenerator-runtime": "^0.13.2" + } + }, + "node_modules/@babel/template": { + "version": "7.20.7", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.20.7.tgz", + "integrity": "sha512-8SegXApWe6VoNw0r9JHpSteLKTpTiLZ4rMlGIm9JQ18KiCtyQiAMEazujAHrUS5flrcqYZa75ukev3P6QmUwUw==", + "dev": true, + "dependencies": { + "@babel/code-frame": "^7.18.6", + "@babel/parser": "^7.20.7", + "@babel/types": "^7.20.7" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/traverse": { + "version": "7.20.13", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.20.13.tgz", + "integrity": "sha512-kMJXfF0T6DIS9E8cgdLCSAL+cuCK+YEZHWiLK0SXpTo8YRj5lpJu3CDNKiIBCne4m9hhTIqUg6SYTAI39tAiVQ==", + "dev": true, + "dependencies": { + "@babel/code-frame": "^7.18.6", + "@babel/generator": "^7.20.7", + "@babel/helper-environment-visitor": "^7.18.9", + "@babel/helper-function-name": "^7.19.0", + "@babel/helper-hoist-variables": "^7.18.6", + "@babel/helper-split-export-declaration": "^7.18.6", + "@babel/parser": "^7.20.13", + "@babel/types": "^7.20.7", + "debug": "^4.1.0", + "globals": "^11.1.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/types": { + "version": "7.20.7", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.20.7.tgz", + "integrity": "sha512-69OnhBxSSgK0OzTJai4kyPDiKTIe3j+ctaHdIGVbRahTLAT7L3R9oeXHC2aVSuGYt3cVnoAMDmOCgJ2yaiLMvg==", + "dev": true, + "dependencies": { + "@babel/helper-string-parser": "^7.19.4", + "@babel/helper-validator-identifier": "^7.19.1", + "to-fast-properties": "^2.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@cookbook/solid-intl": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/@cookbook/solid-intl/-/solid-intl-0.1.2.tgz", + "integrity": "sha512-mrnm8MJ+rskAF0txqs0mxBU4tra3HAnoxjqlcGMH3Bnfkgj90HycpNND7bLYeCMsTQm93Kz5C1AidbHwoSYppA==", + "dependencies": { + "@formatjs/intl": "^2.6.3" + }, + "peerDependencies": { + "solid-js": ">=1.0.0" + } + }, + "node_modules/@esbuild/android-arm": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.16.17.tgz", + "integrity": "sha512-N9x1CMXVhtWEAMS7pNNONyA14f71VPQN9Cnavj1XQh6T7bskqiLLrSca4O0Vr8Wdcga943eThxnVp3JLnBMYtw==", + "cpu": [ + "arm" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-arm64": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.16.17.tgz", + "integrity": "sha512-MIGl6p5sc3RDTLLkYL1MyL8BMRN4tLMRCn+yRJJmEDvYZ2M7tmAf80hx1kbNEUX2KJ50RRtxZ4JHLvCfuB6kBg==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/android-x64": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.16.17.tgz", + "integrity": "sha512-a3kTv3m0Ghh4z1DaFEuEDfz3OLONKuFvI4Xqczqx4BqLyuFaFkuaG4j2MtA6fuWEFeC5x9IvqnX7drmRq/fyAQ==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/darwin-arm64": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.16.17.tgz", + "integrity": "sha512-/2agbUEfmxWHi9ARTX6OQ/KgXnOWfsNlTeLcoV7HSuSTv63E4DqtAc+2XqGw1KHxKMHGZgbVCZge7HXWX9Vn+w==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/darwin-x64": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.16.17.tgz", + "integrity": "sha512-2By45OBHulkd9Svy5IOCZt376Aa2oOkiE9QWUK9fe6Tb+WDr8hXL3dpqi+DeLiMed8tVXspzsTAvd0jUl96wmg==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/freebsd-arm64": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.16.17.tgz", + "integrity": "sha512-mt+cxZe1tVx489VTb4mBAOo2aKSnJ33L9fr25JXpqQqzbUIw/yzIzi+NHwAXK2qYV1lEFp4OoVeThGjUbmWmdw==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/freebsd-x64": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.16.17.tgz", + "integrity": "sha512-8ScTdNJl5idAKjH8zGAsN7RuWcyHG3BAvMNpKOBaqqR7EbUhhVHOqXRdL7oZvz8WNHL2pr5+eIT5c65kA6NHug==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-arm": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.16.17.tgz", + "integrity": "sha512-iihzrWbD4gIT7j3caMzKb/RsFFHCwqqbrbH9SqUSRrdXkXaygSZCZg1FybsZz57Ju7N/SHEgPyaR0LZ8Zbe9gQ==", + "cpu": [ + "arm" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-arm64": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.16.17.tgz", + "integrity": "sha512-7S8gJnSlqKGVJunnMCrXHU9Q8Q/tQIxk/xL8BqAP64wchPCTzuM6W3Ra8cIa1HIflAvDnNOt2jaL17vaW+1V0g==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-ia32": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.16.17.tgz", + "integrity": "sha512-kiX69+wcPAdgl3Lonh1VI7MBr16nktEvOfViszBSxygRQqSpzv7BffMKRPMFwzeJGPxcio0pdD3kYQGpqQ2SSg==", + "cpu": [ + "ia32" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-loong64": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.16.17.tgz", + "integrity": "sha512-dTzNnQwembNDhd654cA4QhbS9uDdXC3TKqMJjgOWsC0yNCbpzfWoXdZvp0mY7HU6nzk5E0zpRGGx3qoQg8T2DQ==", + "cpu": [ + "loong64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-mips64el": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.16.17.tgz", + "integrity": "sha512-ezbDkp2nDl0PfIUn0CsQ30kxfcLTlcx4Foz2kYv8qdC6ia2oX5Q3E/8m6lq84Dj/6b0FrkgD582fJMIfHhJfSw==", + "cpu": [ + "mips64el" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-ppc64": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.16.17.tgz", + "integrity": "sha512-dzS678gYD1lJsW73zrFhDApLVdM3cUF2MvAa1D8K8KtcSKdLBPP4zZSLy6LFZ0jYqQdQ29bjAHJDgz0rVbLB3g==", + "cpu": [ + "ppc64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-riscv64": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.16.17.tgz", + "integrity": "sha512-ylNlVsxuFjZK8DQtNUwiMskh6nT0vI7kYl/4fZgV1llP5d6+HIeL/vmmm3jpuoo8+NuXjQVZxmKuhDApK0/cKw==", + "cpu": [ + "riscv64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-s390x": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.16.17.tgz", + "integrity": "sha512-gzy7nUTO4UA4oZ2wAMXPNBGTzZFP7mss3aKR2hH+/4UUkCOyqmjXiKpzGrY2TlEUhbbejzXVKKGazYcQTZWA/w==", + "cpu": [ + "s390x" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/linux-x64": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.16.17.tgz", + "integrity": "sha512-mdPjPxfnmoqhgpiEArqi4egmBAMYvaObgn4poorpUaqmvzzbvqbowRllQ+ZgzGVMGKaPkqUmPDOOFQRUFDmeUw==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/netbsd-x64": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.16.17.tgz", + "integrity": "sha512-/PzmzD/zyAeTUsduZa32bn0ORug+Jd1EGGAUJvqfeixoEISYpGnAezN6lnJoskauoai0Jrs+XSyvDhppCPoKOA==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/openbsd-x64": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.16.17.tgz", + "integrity": "sha512-2yaWJhvxGEz2RiftSk0UObqJa/b+rIAjnODJgv2GbGGpRwAfpgzyrg1WLK8rqA24mfZa9GvpjLcBBg8JHkoodg==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/sunos-x64": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.16.17.tgz", + "integrity": "sha512-xtVUiev38tN0R3g8VhRfN7Zl42YCJvyBhRKw1RJjwE1d2emWTVToPLNEQj/5Qxc6lVFATDiy6LjVHYhIPrLxzw==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "sunos" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-arm64": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.16.17.tgz", + "integrity": "sha512-ga8+JqBDHY4b6fQAmOgtJJue36scANy4l/rL97W+0wYmijhxKetzZdKOJI7olaBaMhWt8Pac2McJdZLxXWUEQw==", + "cpu": [ + "arm64" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-ia32": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.16.17.tgz", + "integrity": "sha512-WnsKaf46uSSF/sZhwnqE4L/F89AYNMiD4YtEcYekBt9Q7nj0DiId2XH2Ng2PHM54qi5oPrQ8luuzGszqi/veig==", + "cpu": [ + "ia32" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@esbuild/win32-x64": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.16.17.tgz", + "integrity": "sha512-y+EHuSchhL7FjHgvQL/0fnnFmO4T1bhvWANX6gcnqTjtnKWbTvUMCpGnv2+t+31d7RzyEAYAd4u2fnIhHL6N/Q==", + "cpu": [ + "x64" + ], + "dev": true, + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=12" + } + }, + "node_modules/@formatjs/cli": { + "version": "6.0.4", + "resolved": "https://registry.npmjs.org/@formatjs/cli/-/cli-6.0.4.tgz", + "integrity": "sha512-ivb+uUcYmHnffBkXM7OM4NDofxyfnVvW5G52p+M9Cg3DGMz3wVBm3TwW3SXgGGTft7CMWHeGQGXjxTOwBYKeEA==", + "dev": true, + "bin": { + "formatjs": "bin/formatjs" + }, + "engines": { + "node": ">= 16" + }, + "peerDependencies": { + "@vue/compiler-sfc": "^3.2.34" + }, + "peerDependenciesMeta": { + "@vue/compiler-sfc": { + "optional": true + } + } + }, + "node_modules/@formatjs/ecma402-abstract": { + "version": "1.14.3", + "resolved": "https://registry.npmjs.org/@formatjs/ecma402-abstract/-/ecma402-abstract-1.14.3.tgz", + "integrity": "sha512-SlsbRC/RX+/zg4AApWIFNDdkLtFbkq3LNoZWXZCE/nHVKqoIJyaoQyge/I0Y38vLxowUn9KTtXgusLD91+orbg==", + "dependencies": { + "@formatjs/intl-localematcher": "0.2.32", + "tslib": "^2.4.0" + } + }, + "node_modules/@formatjs/fast-memoize": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/@formatjs/fast-memoize/-/fast-memoize-2.0.1.tgz", + "integrity": "sha512-M2GgV+qJn5WJQAYewz7q2Cdl6fobQa69S1AzSM2y0P68ZDbK5cWrJIcPCO395Of1ksftGZoOt4LYCO/j9BKBSA==", + "dependencies": { + "tslib": "^2.4.0" + } + }, + "node_modules/@formatjs/icu-messageformat-parser": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/@formatjs/icu-messageformat-parser/-/icu-messageformat-parser-2.3.0.tgz", + "integrity": "sha512-xqtlqYAbfJDF4b6e4O828LBNOWXrFcuYadqAbYORlDRwhyJ2bH+xpUBPldZbzRGUN2mxlZ4Ykhm7jvERtmI8NQ==", + "dependencies": { + "@formatjs/ecma402-abstract": "1.14.3", + "@formatjs/icu-skeleton-parser": "1.3.18", + "tslib": "^2.4.0" + } + }, + "node_modules/@formatjs/icu-skeleton-parser": { + "version": "1.3.18", + "resolved": "https://registry.npmjs.org/@formatjs/icu-skeleton-parser/-/icu-skeleton-parser-1.3.18.tgz", + "integrity": "sha512-ND1ZkZfmLPcHjAH1sVpkpQxA+QYfOX3py3SjKWMUVGDow18gZ0WPqz3F+pJLYQMpS2LnnQ5zYR2jPVYTbRwMpg==", + "dependencies": { + "@formatjs/ecma402-abstract": "1.14.3", + "tslib": "^2.4.0" + } + }, + "node_modules/@formatjs/intl": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/@formatjs/intl/-/intl-2.6.9.tgz", + "integrity": "sha512-EtcMZ9O24YSASu/jGOaTQtArx7XROjlKiO4KmkxJ/3EyAQLCr5hrS+KKvNud0a7GIwBucOb3IFrZ7WiSm2A/Cw==", + "dependencies": { + "@formatjs/ecma402-abstract": "1.14.3", + "@formatjs/fast-memoize": "2.0.1", + "@formatjs/icu-messageformat-parser": "2.3.0", + "@formatjs/intl-displaynames": "6.2.6", + "@formatjs/intl-listformat": "7.1.9", + "intl-messageformat": "10.3.3", + "tslib": "^2.4.0" + }, + "peerDependencies": { + "typescript": "^4.7" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@formatjs/intl-displaynames": { + "version": "6.2.6", + "resolved": "https://registry.npmjs.org/@formatjs/intl-displaynames/-/intl-displaynames-6.2.6.tgz", + "integrity": "sha512-scf5AQTk9EjpvPhboo5sizVOvidTdMOnajv9z+0cejvl7JNl9bl/aMrNBgC72UH+bP3l45usPUKAGskV6sNIrA==", + "dependencies": { + "@formatjs/ecma402-abstract": "1.14.3", + "@formatjs/intl-localematcher": "0.2.32", + "tslib": "^2.4.0" + } + }, + "node_modules/@formatjs/intl-listformat": { + "version": "7.1.9", + "resolved": "https://registry.npmjs.org/@formatjs/intl-listformat/-/intl-listformat-7.1.9.tgz", + "integrity": "sha512-5YikxwRqRXTVWVujhswDOTCq6gs+m9IcNbNZLa6FLtyBStAjEsuE2vAU+lPsbz9ZTST57D5fodjIh2JXT6sMWQ==", + "dependencies": { + "@formatjs/ecma402-abstract": "1.14.3", + "@formatjs/intl-localematcher": "0.2.32", + "tslib": "^2.4.0" + } + }, + "node_modules/@formatjs/intl-localematcher": { + "version": "0.2.32", + "resolved": "https://registry.npmjs.org/@formatjs/intl-localematcher/-/intl-localematcher-0.2.32.tgz", + "integrity": "sha512-k/MEBstff4sttohyEpXxCmC3MqbUn9VvHGlZ8fauLzkbwXmVrEeyzS+4uhrvAk9DWU9/7otYWxyDox4nT/KVLQ==", + "dependencies": { + "tslib": "^2.4.0" + } + }, + "node_modules/@jridgewell/gen-mapping": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.1.1.tgz", + "integrity": "sha512-sQXCasFk+U8lWYEe66WxRDOE9PjVz4vSM51fTu3Hw+ClTpUSQb718772vH3pyS5pShp6lvQM7SxgIDXXXmOX7w==", + "dev": true, + "dependencies": { + "@jridgewell/set-array": "^1.0.0", + "@jridgewell/sourcemap-codec": "^1.4.10" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/resolve-uri": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.0.tgz", + "integrity": "sha512-F2msla3tad+Mfht5cJq7LSXcdudKTWCVYUgw6pLFOOHSTtZlj6SWNYAp+AhuqLmWdBO2X5hPrLcu8cVP8fy28w==", + "dev": true, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/set-array": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@jridgewell/set-array/-/set-array-1.1.2.tgz", + "integrity": "sha512-xnkseuNADM0gt2bs+BvhO0p78Mk762YnZdsuzFV018NoG1Sj1SCQvpSqa7XUaTam5vAGasABV9qXASMKnFMwMw==", + "dev": true, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@jridgewell/sourcemap-codec": { + "version": "1.4.14", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.14.tgz", + "integrity": "sha512-XPSJHWmi394fuUuzDnGz1wiKqWfo1yXecHQMRf2l6hztTO+nPru658AyDngaBe7isIxEkRsPR3FZh+s7iVa4Uw==", + "dev": true + }, + "node_modules/@jridgewell/trace-mapping": { + "version": "0.3.17", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.17.tgz", + "integrity": "sha512-MCNzAp77qzKca9+W/+I0+sEpaUnZoeasnghNeVc41VZCEKaCH73Vq3BZZ/SzWIgrqE4H4ceI+p+b6C0mHf9T4g==", + "dev": true, + "dependencies": { + "@jridgewell/resolve-uri": "3.1.0", + "@jridgewell/sourcemap-codec": "1.4.14" + } + }, + "node_modules/@jukben/emoji-search": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/@jukben/emoji-search/-/emoji-search-2.0.1.tgz", + "integrity": "sha512-jXVcJGTBl+uOsGld+6J+hcHlRt3Vhm9ffvkrb1IeSVXuFCuyklY2XPI2wvSHG1uMGXfgmKbuUe1MCh1ZV3CXNg==", + "dependencies": { + "@babel/runtime": "7.5.5", + "emojilib": "2.4.0", + "match-sorter": "4.0.0" + } + }, + "node_modules/@noble/hashes": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/@noble/hashes/-/hashes-1.2.0.tgz", + "integrity": "sha512-FZfhjEDbT5GRswV3C6uvLPHMiVD6lQBmpoX5+eSiPaMTXte/IKqI5dykDxzZB/WBeK/CDuQRBWarPdi3FNY2zQ==", + "funding": [ + { + "type": "individual", + "url": "https://paulmillr.com/funding/" + } + ] + }, + "node_modules/@noble/secp256k1": { + "version": "1.7.1", + "resolved": "https://registry.npmjs.org/@noble/secp256k1/-/secp256k1-1.7.1.tgz", + "integrity": "sha512-hOUk6AyBFmqVrv7k5WAw/LpszxVbj9gGN4JRkIX52fdFAj1UA61KXmZDvqVEm+pOyec3+fIeZB02LYa/pWOArw==", + "funding": [ + { + "type": "individual", + "url": "https://paulmillr.com/funding/" + } + ] + }, + "node_modules/@picocss/pico": { + "version": "1.5.7", + "resolved": "https://registry.npmjs.org/@picocss/pico/-/pico-1.5.7.tgz", + "integrity": "sha512-RygdXNlSXieAs9jMw/AeqA1ki1kldgEYbRn8BnYZIPfRTM5NWZ4uVzMK6uMPhYlRjoT5wD/OplZvIefnCqyDCQ==" + }, + "node_modules/@scure/base": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@scure/base/-/base-1.1.1.tgz", + "integrity": "sha512-ZxOhsSyxYwLJj3pLZCefNitxsj093tb2vq90mp2txoYeBqbcjDjqFhyM8eUjq/uFm6zJ+mUuqxlS2FkuSY1MTA==", + "funding": [ + { + "type": "individual", + "url": "https://paulmillr.com/funding/" + } + ] + }, + "node_modules/@scure/bip32": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/@scure/bip32/-/bip32-1.1.4.tgz", + "integrity": "sha512-m925ACYK0wPELsF7Z/VdLGmKj1StIeHraPMYB9xiAFiq/PnvqWd/99I0TQ2OZhjjlMDsDJeZlyXMWi0beaA7NA==", + "funding": [ + { + "type": "individual", + "url": "https://paulmillr.com/funding/" + } + ], + "dependencies": { + "@noble/hashes": "~1.2.0", + "@noble/secp256k1": "~1.7.0", + "@scure/base": "~1.1.0" + } + }, + "node_modules/@scure/bip39": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/@scure/bip39/-/bip39-1.1.1.tgz", + "integrity": "sha512-t+wDck2rVkh65Hmv280fYdVdY25J9YeEUIgn2LG1WM6gxFkGzcksoDiUkWVpVp3Oex9xGC68JU2dSbUfwZ2jPg==", + "funding": [ + { + "type": "individual", + "url": "https://paulmillr.com/funding/" + } + ], + "dependencies": { + "@noble/hashes": "~1.2.0", + "@scure/base": "~1.1.0" + } + }, + "node_modules/@solidjs/router": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/@solidjs/router/-/router-0.7.0.tgz", + "integrity": "sha512-8HI84twe5FjYRebSLMAhtkL9bRuTDIlxJK56kjfjU9WKGoUCTaWpCnkuj8Hqde1bWZ0X+GOZxKDfNkn1CjtjxA==", + "peerDependencies": { + "solid-js": "^1.5.3" + } + }, + "node_modules/@thisbeyond/solid-select": { + "version": "0.13.0", + "resolved": "https://registry.npmjs.org/@thisbeyond/solid-select/-/solid-select-0.13.0.tgz", + "integrity": "sha512-eION+Xf8TGLs1NZrvRo1NRKOl4plYMbY7UswHhh5bEUY8oMltjrBhUWF0hzaFViEc1zZpkCQyafaD89iofG6Tg==", + "peerDependencies": { + "solid-js": "^1.5" + } + }, + "node_modules/@types/dompurify": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/@types/dompurify/-/dompurify-2.4.0.tgz", + "integrity": "sha512-IDBwO5IZhrKvHFUl+clZxgf3hn2b/lU6H1KaBShPkQyGJUQ0xwebezIPSuiyGwfz1UzJWQl4M7BDxtHtCCPlTg==", + "dependencies": { + "@types/trusted-types": "*" + } + }, + "node_modules/@types/trusted-types": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/@types/trusted-types/-/trusted-types-2.0.3.tgz", + "integrity": "sha512-NfQ4gyz38SL8sDNrSixxU2Os1a5xcdFxipAFxYEuLUlvU2uDwS4NUpsImcf1//SlWItCVMMLiylsxbmNMToV/g==" + }, + "node_modules/abort-controller": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/abort-controller/-/abort-controller-3.0.0.tgz", + "integrity": "sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==", + "dependencies": { + "event-target-shim": "^5.0.0" + }, + "engines": { + "node": ">=6.5" + } + }, + "node_modules/ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, + "dependencies": { + "color-convert": "^1.9.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/anymatch": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz", + "integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==", + "dependencies": { + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/babel-plugin-jsx-dom-expressions": { + "version": "0.35.15", + "resolved": "https://registry.npmjs.org/babel-plugin-jsx-dom-expressions/-/babel-plugin-jsx-dom-expressions-0.35.15.tgz", + "integrity": "sha512-33GQnanjYKefOTO2lQK6EaKXPJ1W8vtzvBneGfhKaOZHQJLqe61P93jP0TLTz67sqsA0m1ph1cNdGpLc/Nx2Xg==", + "dev": true, + "dependencies": { + "@babel/helper-module-imports": "7.18.6", + "@babel/plugin-syntax-jsx": "^7.18.6", + "@babel/types": "^7.20.7", + "html-entities": "2.3.3", + "validate-html-nesting": "^1.2.0" + }, + "peerDependencies": { + "@babel/core": "^7.20.12" + } + }, + "node_modules/babel-preset-solid": { + "version": "1.6.10", + "resolved": "https://registry.npmjs.org/babel-preset-solid/-/babel-preset-solid-1.6.10.tgz", + "integrity": "sha512-qBLjzeWmgY5jX11sJg/lriXABYdClfJrJJrIHaT6G5EuGhxhm6jn7XjqXjLBZHBgy5n/Z+iqJ5YfQj8KG2jKTA==", + "dev": true, + "dependencies": { + "babel-plugin-jsx-dom-expressions": "^0.35.15" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/binary-extensions": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", + "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", + "engines": { + "node": ">=8" + } + }, + "node_modules/boolbase": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz", + "integrity": "sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww==" + }, + "node_modules/braces": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", + "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "dependencies": { + "fill-range": "^7.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/browserslist": { + "version": "4.21.5", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.21.5.tgz", + "integrity": "sha512-tUkiguQGW7S3IhB7N+c2MV/HZPSCPAAiYBZXLsBhFB/PCy6ZKKsZrmBayHV9fdGV/ARIfJ14NkxKzRDjvp7L6w==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + } + ], + "dependencies": { + "caniuse-lite": "^1.0.30001449", + "electron-to-chromium": "^1.4.284", + "node-releases": "^2.0.8", + "update-browserslist-db": "^1.0.10" + }, + "bin": { + "browserslist": "cli.js" + }, + "engines": { + "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" + } + }, + "node_modules/caniuse-lite": { + "version": "1.0.30001449", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001449.tgz", + "integrity": "sha512-CPB+UL9XMT/Av+pJxCKGhdx+yg1hzplvFJQlJ2n68PyQGMz9L/E2zCyLdOL8uasbouTUgnPl+y0tccI/se+BEw==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/caniuse-lite" + } + ] + }, + "node_modules/chalk": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", + "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", + "dev": true, + "dependencies": { + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/cheerio": { + "version": "1.0.0-rc.11", + "resolved": "https://registry.npmjs.org/cheerio/-/cheerio-1.0.0-rc.11.tgz", + "integrity": "sha512-bQwNaDIBKID5ts/DsdhxrjqFXYfLw4ste+wMKqWA8DyKcS4qwsPP4Bk8ZNaTJjvpiX/qW3BT4sU7d6Bh5i+dag==", + "dependencies": { + "cheerio-select": "^2.1.0", + "dom-serializer": "^2.0.0", + "domhandler": "^5.0.3", + "domutils": "^3.0.1", + "htmlparser2": "^8.0.1", + "parse5": "^7.0.0", + "parse5-htmlparser2-tree-adapter": "^7.0.0", + "tslib": "^2.4.0" + }, + "engines": { + "node": ">= 6" + }, + "funding": { + "url": "https://github.com/cheeriojs/cheerio?sponsor=1" + } + }, + "node_modules/cheerio-select": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/cheerio-select/-/cheerio-select-2.1.0.tgz", + "integrity": "sha512-9v9kG0LvzrlcungtnJtpGNxY+fzECQKhK4EGJX2vByejiMX84MFNQw4UxPJl3bFbTMw+Dfs37XaIkCwTZfLh4g==", + "dependencies": { + "boolbase": "^1.0.0", + "css-select": "^5.1.0", + "css-what": "^6.1.0", + "domelementtype": "^2.3.0", + "domhandler": "^5.0.3", + "domutils": "^3.0.1" + }, + "funding": { + "url": "https://github.com/sponsors/fb55" + } + }, + "node_modules/chokidar": { + "version": "3.5.3", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.3.tgz", + "integrity": "sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==", + "funding": [ + { + "type": "individual", + "url": "https://paulmillr.com/funding/" + } + ], + "dependencies": { + "anymatch": "~3.1.2", + "braces": "~3.0.2", + "glob-parent": "~5.1.2", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.6.0" + }, + "engines": { + "node": ">= 8.10.0" + }, + "optionalDependencies": { + "fsevents": "~2.3.2" + } + }, + "node_modules/color-convert": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", + "dev": true, + "dependencies": { + "color-name": "1.1.3" + } + }, + "node_modules/color-name": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==", + "dev": true + }, + "node_modules/convert-source-map": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.9.0.tgz", + "integrity": "sha512-ASFBup0Mz1uyiIjANan1jzLQami9z1PoYSZCiiYW2FczPbenXc45FZdBZLzOT+r6+iciuEModtmCti+hjaAk0A==", + "dev": true + }, + "node_modules/cross-fetch": { + "version": "3.1.5", + "resolved": "https://registry.npmjs.org/cross-fetch/-/cross-fetch-3.1.5.tgz", + "integrity": "sha512-lvb1SBsI0Z7GDwmuid+mU3kWVBwTVUbe7S0H52yaaAdQOXq2YktTCZdlAcNKFzE6QtRz0snpw9bNiPeOIkkQvw==", + "dependencies": { + "node-fetch": "2.6.7" + } + }, + "node_modules/css-select": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/css-select/-/css-select-5.1.0.tgz", + "integrity": "sha512-nwoRF1rvRRnnCqqY7updORDsuqKzqYJ28+oSMaJMMgOauh3fvwHqMS7EZpIPqK8GL+g9mKxF1vP/ZjSeNjEVHg==", + "dependencies": { + "boolbase": "^1.0.0", + "css-what": "^6.1.0", + "domhandler": "^5.0.2", + "domutils": "^3.0.1", + "nth-check": "^2.0.1" + }, + "funding": { + "url": "https://github.com/sponsors/fb55" + } + }, + "node_modules/css-what": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/css-what/-/css-what-6.1.0.tgz", + "integrity": "sha512-HTUrgRJ7r4dsZKU6GjmpfRK1O76h97Z8MfS1G0FozR+oF2kG6Vfe8JE6zwrkbxigziPHinCJ+gCPjA9EaBDtRw==", + "engines": { + "node": ">= 6" + }, + "funding": { + "url": "https://github.com/sponsors/fb55" + } + }, + "node_modules/csstype": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.1.tgz", + "integrity": "sha512-DJR/VvkAvSZW9bTouZue2sSxDwdTN92uHjqeKVm+0dAqdfNykRzQ95tay8aXMBAAPpUiq4Qcug2L7neoRh2Egw==" + }, + "node_modules/debug": { + "version": "4.3.4", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/dom-serializer": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-2.0.0.tgz", + "integrity": "sha512-wIkAryiqt/nV5EQKqQpo3SToSOV9J0DnbJqwK7Wv/Trc92zIAYZ4FlMu+JPFW1DfGFt81ZTCGgDEabffXeLyJg==", + "dependencies": { + "domelementtype": "^2.3.0", + "domhandler": "^5.0.2", + "entities": "^4.2.0" + }, + "funding": { + "url": "https://github.com/cheeriojs/dom-serializer?sponsor=1" + } + }, + "node_modules/domelementtype": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-2.3.0.tgz", + "integrity": "sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/fb55" + } + ] + }, + "node_modules/domhandler": { + "version": "5.0.3", + "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-5.0.3.tgz", + "integrity": "sha512-cgwlv/1iFQiFnU96XXgROh8xTeetsnJiDsTc7TYCLFd9+/WNkIqPTxiM/8pSd8VIrhXGTf1Ny1q1hquVqDJB5w==", + "dependencies": { + "domelementtype": "^2.3.0" + }, + "engines": { + "node": ">= 4" + }, + "funding": { + "url": "https://github.com/fb55/domhandler?sponsor=1" + } + }, + "node_modules/dompurify": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/dompurify/-/dompurify-3.0.0.tgz", + "integrity": "sha512-0g/yr2IJn4nTbxwL785YxS7/AvvgGFJw6LLWP+BzWzB1+BYOqPUT9Hy0rXrZh5HLdHnxH72aDdzvC9SdTjsuaA==" + }, + "node_modules/domutils": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/domutils/-/domutils-3.1.0.tgz", + "integrity": "sha512-H78uMmQtI2AhgDJjWeQmHwJJ2bLPD3GMmO7Zja/ZZh84wkm+4ut+IUnUdRa8uCGX88DiVx1j6FRe1XfxEgjEZA==", + "dependencies": { + "dom-serializer": "^2.0.0", + "domelementtype": "^2.3.0", + "domhandler": "^5.0.3" + }, + "funding": { + "url": "https://github.com/fb55/domutils?sponsor=1" + } + }, + "node_modules/electron-to-chromium": { + "version": "1.4.284", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.284.tgz", + "integrity": "sha512-M8WEXFuKXMYMVr45fo8mq0wUrrJHheiKZf6BArTKk9ZBYCKJEOU5H8cdWgDT+qCVZf7Na4lVUaZsA+h6uA9+PA==", + "dev": true + }, + "node_modules/emojilib": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/emojilib/-/emojilib-2.4.0.tgz", + "integrity": "sha512-5U0rVMU5Y2n2+ykNLQqMoqklN9ICBT/KsvC1Gz6vqHbz2AXXGkG+Pm5rMWk/8Vjrr/mY9985Hi8DYzn1F09Nyw==" + }, + "node_modules/entities": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/entities/-/entities-4.5.0.tgz", + "integrity": "sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw==", + "engines": { + "node": ">=0.12" + }, + "funding": { + "url": "https://github.com/fb55/entities?sponsor=1" + } + }, + "node_modules/esbuild": { + "version": "0.16.17", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.16.17.tgz", + "integrity": "sha512-G8LEkV0XzDMNwXKgM0Jwu3nY3lSTwSGY6XbxM9cr9+s0T/qSV1q1JVPBGzm3dcjhCic9+emZDmMffkwgPeOeLg==", + "dev": true, + "hasInstallScript": true, + "bin": { + "esbuild": "bin/esbuild" + }, + "engines": { + "node": ">=12" + }, + "optionalDependencies": { + "@esbuild/android-arm": "0.16.17", + "@esbuild/android-arm64": "0.16.17", + "@esbuild/android-x64": "0.16.17", + "@esbuild/darwin-arm64": "0.16.17", + "@esbuild/darwin-x64": "0.16.17", + "@esbuild/freebsd-arm64": "0.16.17", + "@esbuild/freebsd-x64": "0.16.17", + "@esbuild/linux-arm": "0.16.17", + "@esbuild/linux-arm64": "0.16.17", + "@esbuild/linux-ia32": "0.16.17", + "@esbuild/linux-loong64": "0.16.17", + "@esbuild/linux-mips64el": "0.16.17", + "@esbuild/linux-ppc64": "0.16.17", + "@esbuild/linux-riscv64": "0.16.17", + "@esbuild/linux-s390x": "0.16.17", + "@esbuild/linux-x64": "0.16.17", + "@esbuild/netbsd-x64": "0.16.17", + "@esbuild/openbsd-x64": "0.16.17", + "@esbuild/sunos-x64": "0.16.17", + "@esbuild/win32-arm64": "0.16.17", + "@esbuild/win32-ia32": "0.16.17", + "@esbuild/win32-x64": "0.16.17" + } + }, + "node_modules/escalade": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", + "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/event-target-shim": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/event-target-shim/-/event-target-shim-5.0.1.tgz", + "integrity": "sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==", + "engines": { + "node": ">=6" + } + }, + "node_modules/fill-range": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", + "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "dependencies": { + "to-regex-range": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/fsevents": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", + "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", + "hasInstallScript": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "node_modules/function-bind": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", + "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", + "dev": true + }, + "node_modules/gensync": { + "version": "1.0.0-beta.2", + "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", + "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dependencies": { + "is-glob": "^4.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/globals": { + "version": "11.12.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", + "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/has": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", + "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", + "dev": true, + "dependencies": { + "function-bind": "^1.1.1" + }, + "engines": { + "node": ">= 0.4.0" + } + }, + "node_modules/has-flag": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", + "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/html-entities": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/html-entities/-/html-entities-2.3.3.tgz", + "integrity": "sha512-DV5Ln36z34NNTDgnz0EWGBLZENelNAtkiFA4kyNOG2tDI6Mz1uSWiq1wAKdyjnJwyDiDO7Fa2SO1CTxPXL8VxA==", + "dev": true + }, + "node_modules/htmlparser2": { + "version": "8.0.2", + "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-8.0.2.tgz", + "integrity": "sha512-GYdjWKDkbRLkZ5geuHs5NY1puJ+PXwP7+fHPRz06Eirsb9ugf6d8kkXav6ADhcODhFFPMIXyxkxSuMf3D6NCFA==", + "funding": [ + "https://github.com/fb55/htmlparser2?sponsor=1", + { + "type": "github", + "url": "https://github.com/sponsors/fb55" + } + ], + "dependencies": { + "domelementtype": "^2.3.0", + "domhandler": "^5.0.3", + "domutils": "^3.0.1", + "entities": "^4.4.0" + } + }, + "node_modules/immutable": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/immutable/-/immutable-4.2.2.tgz", + "integrity": "sha512-fTMKDwtbvO5tldky9QZ2fMX7slR0mYpY5nbnFWYp0fOzDhHqhgIw9KoYgxLWsoNTS9ZHGauHj18DTyEw6BK3Og==" + }, + "node_modules/intl-messageformat": { + "version": "10.3.3", + "resolved": "https://registry.npmjs.org/intl-messageformat/-/intl-messageformat-10.3.3.tgz", + "integrity": "sha512-un/f07/g2e/3Q8e1ghDKET+el22Bi49M7O/rHxd597R+oLpPOMykSv5s51cABVfu3FZW+fea4hrzf2MHu1W4hw==", + "dependencies": { + "@formatjs/ecma402-abstract": "1.14.3", + "@formatjs/fast-memoize": "2.0.1", + "@formatjs/icu-messageformat-parser": "2.3.0", + "tslib": "^2.4.0" + } + }, + "node_modules/is-binary-path": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", + "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", + "dependencies": { + "binary-extensions": "^2.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/is-core-module": { + "version": "2.11.0", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.11.0.tgz", + "integrity": "sha512-RRjxlvLDkD1YJwDbroBHMb+cukurkDWNyHx7D3oNB5x9rb5ogcksMC5wHCadcXoo67gVr/+3GFySh3134zi6rw==", + "dev": true, + "dependencies": { + "has": "^1.0.3" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/is-what": { + "version": "4.1.8", + "resolved": "https://registry.npmjs.org/is-what/-/is-what-4.1.8.tgz", + "integrity": "sha512-yq8gMao5upkPoGEU9LsB2P+K3Kt8Q3fQFCGyNCWOAnJAMzEXVV9drYb0TXr42TTliLLhKIBvulgAXgtLLnwzGA==", + "dev": true, + "engines": { + "node": ">=12.13" + }, + "funding": { + "url": "https://github.com/sponsors/mesqueeb" + } + }, + "node_modules/js-tokens": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", + "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", + "dev": true + }, + "node_modules/jsesc": { + "version": "2.5.2", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-2.5.2.tgz", + "integrity": "sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==", + "dev": true, + "bin": { + "jsesc": "bin/jsesc" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/json5": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz", + "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==", + "dev": true, + "bin": { + "json5": "lib/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/link-preview-js": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/link-preview-js/-/link-preview-js-3.0.4.tgz", + "integrity": "sha512-xsuxMigAZd4xmj6BIwMNuQjjpJdh0DWeIo1NXQgaoWSi9Z/dzz/Kxy6vzzsUonFlMTPJ1i0EC8aeOg/xrOMidg==", + "dependencies": { + "abort-controller": "^3.0.0", + "cheerio": "1.0.0-rc.11", + "cross-fetch": "3.1.5", + "url": "0.11.0" + } + }, + "node_modules/lru-cache": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", + "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", + "dev": true, + "dependencies": { + "yallist": "^3.0.2" + } + }, + "node_modules/match-sorter": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/match-sorter/-/match-sorter-4.0.0.tgz", + "integrity": "sha512-E4DWje5l7+VvDUlqnACXy1iecuD6ZNiqUFw/DUYdFQljRIskZVHoT+76lLv5zz2BQOTxF2CUEgP1/Xu9Xn3MyQ==", + "dependencies": { + "remove-accents": "0.4.2" + } + }, + "node_modules/merge-anything": { + "version": "5.1.4", + "resolved": "https://registry.npmjs.org/merge-anything/-/merge-anything-5.1.4.tgz", + "integrity": "sha512-7PWKwGOs5WWcpw+/OvbiFiAvEP6bv/QHiicigpqMGKIqPPAtGhBLR8LFJW+Zu6m9TXiR/a8+AiPlGG0ko1ruoQ==", + "dev": true, + "dependencies": { + "is-what": "^4.1.8" + }, + "engines": { + "node": ">=12.13" + }, + "funding": { + "url": "https://github.com/sponsors/mesqueeb" + } + }, + "node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "node_modules/nanoid": { + "version": "3.3.4", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.4.tgz", + "integrity": "sha512-MqBkQh/OHTS2egovRtLk45wEyNXwF+cokD+1YPf9u5VfJiRdAiRwB2froX5Co9Rh20xs4siNPm8naNotSD6RBw==", + "dev": true, + "bin": { + "nanoid": "bin/nanoid.cjs" + }, + "engines": { + "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" + } + }, + "node_modules/node-fetch": { + "version": "2.6.7", + "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.7.tgz", + "integrity": "sha512-ZjMPFEfVx5j+y2yF35Kzx5sF7kDzxuDj6ziH4FFbOp87zKDZNx8yExJIb05OGF4Nlt9IHFIMBkRl41VdvcNdbQ==", + "dependencies": { + "whatwg-url": "^5.0.0" + }, + "engines": { + "node": "4.x || >=6.0.0" + }, + "peerDependencies": { + "encoding": "^0.1.0" + }, + "peerDependenciesMeta": { + "encoding": { + "optional": true + } + } + }, + "node_modules/node-releases": { + "version": "2.0.9", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.9.tgz", + "integrity": "sha512-2xfmOrRkGogbTK9R6Leda0DGiXeY3p2NJpy4+gNCffdUvV6mdEJnaDEic1i3Ec2djAo8jWYoJMR5PB0MSMpxUA==", + "dev": true + }, + "node_modules/normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/nostr-tools": { + "version": "1.10.1", + "resolved": "https://registry.npmjs.org/nostr-tools/-/nostr-tools-1.10.1.tgz", + "integrity": "sha512-zgTYJeuZQ3CDASsmBEcB5i6V6l0IaA6cjnll6OVik3FoZcvbCaL7yP8I40hYnOIi3KlJykV7jEF9fn8h1NzMnA==", + "dependencies": { + "@noble/hashes": "1.2.0", + "@noble/secp256k1": "1.7.1", + "@scure/base": "1.1.1", + "@scure/bip32": "1.1.4", + "@scure/bip39": "1.1.1" + } + }, + "node_modules/nth-check": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/nth-check/-/nth-check-2.1.1.tgz", + "integrity": "sha512-lqjrjmaOoAnWfMmBPL+XNnynZh2+swxiX3WUE0s4yEHI6m+AwrK2UZOimIRl3X/4QctVqS8AiZjFqyOGrMXb/w==", + "dependencies": { + "boolbase": "^1.0.0" + }, + "funding": { + "url": "https://github.com/fb55/nth-check?sponsor=1" + } + }, + "node_modules/parse5": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/parse5/-/parse5-7.1.2.tgz", + "integrity": "sha512-Czj1WaSVpaoj0wbhMzLmWD69anp2WH7FXMB9n1Sy8/ZFF9jolSQVMu1Ij5WIyGmcBmhk7EOndpO4mIpihVqAXw==", + "dependencies": { + "entities": "^4.4.0" + }, + "funding": { + "url": "https://github.com/inikulin/parse5?sponsor=1" + } + }, + "node_modules/parse5-htmlparser2-tree-adapter": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/parse5-htmlparser2-tree-adapter/-/parse5-htmlparser2-tree-adapter-7.0.0.tgz", + "integrity": "sha512-B77tOZrqqfUfnVcOrUvfdLbz4pu4RopLD/4vmu3HUPswwTA8OH0EMW9BlWR2B0RCoiZRAHEUu7IxeP1Pd1UU+g==", + "dependencies": { + "domhandler": "^5.0.2", + "parse5": "^7.0.0" + }, + "funding": { + "url": "https://github.com/inikulin/parse5?sponsor=1" + } + }, + "node_modules/path-parse": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", + "dev": true + }, + "node_modules/picocolors": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.0.0.tgz", + "integrity": "sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ==", + "dev": true + }, + "node_modules/picomatch": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/postcss": { + "version": "8.4.21", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.21.tgz", + "integrity": "sha512-tP7u/Sn/dVxK2NnruI4H9BG+x+Wxz6oeZ1cJ8P6G/PZY0IKk4k/63TDsQf2kQq3+qoJeLm2kIBUNlZe3zgb4Zg==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/postcss/" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/postcss" + } + ], + "dependencies": { + "nanoid": "^3.3.4", + "picocolors": "^1.0.0", + "source-map-js": "^1.0.2" + }, + "engines": { + "node": "^10 || ^12 || >=14" + } + }, + "node_modules/punycode": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.3.2.tgz", + "integrity": "sha512-RofWgt/7fL5wP1Y7fxE7/EmTLzQVnB0ycyibJ0OOHIlJqTNzglYFxVwETOcIoJqJmpDXJ9xImDv+Fq34F/d4Dw==" + }, + "node_modules/querystring": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/querystring/-/querystring-0.2.0.tgz", + "integrity": "sha512-X/xY82scca2tau62i9mDyU9K+I+djTMUsvwf7xnUX5GLvVzgJybOJf4Y6o9Zx3oJK/LSXg5tTZBjwzqVPaPO2g==", + "deprecated": "The querystring API is considered Legacy. new code should use the URLSearchParams API instead.", + "engines": { + "node": ">=0.4.x" + } + }, + "node_modules/readdirp": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", + "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", + "dependencies": { + "picomatch": "^2.2.1" + }, + "engines": { + "node": ">=8.10.0" + } + }, + "node_modules/regenerator-runtime": { + "version": "0.13.11", + "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.11.tgz", + "integrity": "sha512-kY1AZVr2Ra+t+piVaJ4gxaFaReZVH40AKNo7UCX6W+dEwBo/2oZJzqfuN1qLq1oL45o56cPaTXELwrTh8Fpggg==" + }, + "node_modules/remove-accents": { + "version": "0.4.2", + "resolved": "https://registry.npmjs.org/remove-accents/-/remove-accents-0.4.2.tgz", + "integrity": "sha512-7pXIJqJOq5tFgG1A2Zxti3Ht8jJF337m4sowbuHsW30ZnkQFnDzy9qBNhgzX8ZLW4+UBcXiiR7SwR6pokHsxiA==" + }, + "node_modules/resolve": { + "version": "1.22.1", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.1.tgz", + "integrity": "sha512-nBpuuYuY5jFsli/JIs1oldw6fOQCBioohqWZg/2hiaOybXOft4lonv85uDOKXdf8rhyK159cxU5cDcK/NKk8zw==", + "dev": true, + "dependencies": { + "is-core-module": "^2.9.0", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + }, + "bin": { + "resolve": "bin/resolve" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/rollup": { + "version": "3.12.0", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-3.12.0.tgz", + "integrity": "sha512-4MZ8kA2HNYahIjz63rzrMMRvDqQDeS9LoriJvMuV0V6zIGysP36e9t4yObUfwdT9h/szXoHQideICftcdZklWg==", + "dev": true, + "bin": { + "rollup": "dist/bin/rollup" + }, + "engines": { + "node": ">=14.18.0", + "npm": ">=8.0.0" + }, + "optionalDependencies": { + "fsevents": "~2.3.2" + } + }, + "node_modules/sass": { + "version": "1.58.0", + "resolved": "https://registry.npmjs.org/sass/-/sass-1.58.0.tgz", + "integrity": "sha512-PiMJcP33DdKtZ/1jSjjqVIKihoDc6yWmYr9K/4r3fVVIEDAluD0q7XZiRKrNJcPK3qkLRF/79DND1H5q1LBjgg==", + "dependencies": { + "chokidar": ">=3.0.0 <4.0.0", + "immutable": "^4.0.0", + "source-map-js": ">=0.6.2 <2.0.0" + }, + "bin": { + "sass": "sass.js" + }, + "engines": { + "node": ">=12.0.0" + } + }, + "node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/solid-js": { + "version": "1.6.10", + "resolved": "https://registry.npmjs.org/solid-js/-/solid-js-1.6.10.tgz", + "integrity": "sha512-Sf0e6PQCEFkFtbPq0L+93Ua81YQOefBEbvDJ0YXT92b6Lzw0k7UvzSd2l1BbYM+yzE3UmepU1tyMDc/3nIByjA==", + "dependencies": { + "csstype": "^3.1.0" + } + }, + "node_modules/solid-refresh": { + "version": "0.4.3", + "resolved": "https://registry.npmjs.org/solid-refresh/-/solid-refresh-0.4.3.tgz", + "integrity": "sha512-7+4/gYsVi0BlM4PzT1PU1TB5nW3Hv8FWuB+Kw/ofWui7KQkWBf+dVZOrReQYHEmLCzytHUa2JysUXgzVALJmSw==", + "dev": true, + "dependencies": { + "@babel/generator": "^7.18.7", + "@babel/helper-module-imports": "^7.16.7", + "@babel/types": "^7.18.4" + }, + "peerDependencies": { + "solid-js": "^1.3" + } + }, + "node_modules/source-map-js": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.0.2.tgz", + "integrity": "sha512-R0XvVJ9WusLiqTCEiGCmICCMplcCkIwwR11mOSD9CR5u+IXYdiseeEuXCVAjS54zqwkLcPNnmU4OeJ6tUrWhDw==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "dev": true, + "dependencies": { + "has-flag": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/supports-preserve-symlinks-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", + "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", + "dev": true, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/to-fast-properties": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-2.0.0.tgz", + "integrity": "sha512-/OaKK0xYrs3DmxRYqL/yDc+FxFUVYhDlXMhRmv3z915w2HF1tnN1omB354j8VUGO/hbRzyD6Y3sA7v7GS/ceog==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dependencies": { + "is-number": "^7.0.0" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/tr46": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz", + "integrity": "sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==" + }, + "node_modules/tslib": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.5.0.tgz", + "integrity": "sha512-336iVw3rtn2BUK7ORdIAHTyxHGRIHVReokCR3XjbckJMK7ms8FysBfhLR8IXnAgy7T0PTPNBWKiH514FOW/WSg==" + }, + "node_modules/typescript": { + "version": "4.9.5", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.9.5.tgz", + "integrity": "sha512-1FXk9E2Hm+QzZQ7z+McJiHL4NW1F2EzMu9Nq9i3zAaGqibafqYwCVU6WyWAuyQRRzOlxou8xZSyXLEN8oKj24g==", + "devOptional": true, + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=4.2.0" + } + }, + "node_modules/update-browserslist-db": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.0.10.tgz", + "integrity": "sha512-OztqDenkfFkbSG+tRxBeAnCVPckDBcvibKd35yDONx6OU8N7sqgwc7rCbkJ/WcYtVRZ4ba68d6byhC21GFh7sQ==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" + } + ], + "dependencies": { + "escalade": "^3.1.1", + "picocolors": "^1.0.0" + }, + "bin": { + "browserslist-lint": "cli.js" + }, + "peerDependencies": { + "browserslist": ">= 4.21.0" + } + }, + "node_modules/url": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/url/-/url-0.11.0.tgz", + "integrity": "sha512-kbailJa29QrtXnxgq+DdCEGlbTeYM2eJUxsz6vjZavrCYPMIFHMKQmSKYAIuUK2i7hgPm28a8piX5NTUtM/LKQ==", + "dependencies": { + "punycode": "1.3.2", + "querystring": "0.2.0" + } + }, + "node_modules/validate-html-nesting": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/validate-html-nesting/-/validate-html-nesting-1.2.0.tgz", + "integrity": "sha512-sI65QUd3T/e5wbQkdPKjikFsIVLPIaOQK+9uowPp6/k609SN8hs5eqBLrnN5DeW9Kd932Q4Imo0fzK2dxoOsCA==", + "dev": true + }, + "node_modules/vite": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/vite/-/vite-4.0.4.tgz", + "integrity": "sha512-xevPU7M8FU0i/80DMR+YhgrzR5KS2ORy1B4xcX/cXLsvnUWvfHuqMmVU6N0YiJ4JWGRJJsLCgjEzKjG9/GKoSw==", + "dev": true, + "dependencies": { + "esbuild": "^0.16.3", + "postcss": "^8.4.20", + "resolve": "^1.22.1", + "rollup": "^3.7.0" + }, + "bin": { + "vite": "bin/vite.js" + }, + "engines": { + "node": "^14.18.0 || >=16.0.0" + }, + "optionalDependencies": { + "fsevents": "~2.3.2" + }, + "peerDependencies": { + "@types/node": ">= 14", + "less": "*", + "sass": "*", + "stylus": "*", + "sugarss": "*", + "terser": "^5.4.0" + }, + "peerDependenciesMeta": { + "@types/node": { + "optional": true + }, + "less": { + "optional": true + }, + "sass": { + "optional": true + }, + "stylus": { + "optional": true + }, + "sugarss": { + "optional": true + }, + "terser": { + "optional": true + } + } + }, + "node_modules/vite-plugin-solid": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/vite-plugin-solid/-/vite-plugin-solid-2.5.0.tgz", + "integrity": "sha512-VneGd3RyFJvwaiffsqgymeMaofn0IzQLPwDzafTV2f1agoWeeJlk5VrI5WqT9BTtLe69vNNbCJWqLhHr9fOdDw==", + "dev": true, + "dependencies": { + "@babel/core": "^7.20.5", + "@babel/preset-typescript": "^7.18.6", + "babel-preset-solid": "^1.6.3", + "merge-anything": "^5.1.4", + "solid-refresh": "^0.4.1", + "vitefu": "^0.2.3" + }, + "peerDependencies": { + "solid-js": "^1.3.17 || ^1.4.0 || ^1.5.0 || ^1.6.0", + "vite": "^3.0.0 || ^4.0.0" + } + }, + "node_modules/vitefu": { + "version": "0.2.4", + "resolved": "https://registry.npmjs.org/vitefu/-/vitefu-0.2.4.tgz", + "integrity": "sha512-fanAXjSaf9xXtOOeno8wZXIhgia+CZury481LsDaV++lSvcU2R9Ch2bPh3PYFyoHW+w9LqAeYRISVQjUIew14g==", + "dev": true, + "peerDependencies": { + "vite": "^3.0.0 || ^4.0.0" + }, + "peerDependenciesMeta": { + "vite": { + "optional": true + } + } + }, + "node_modules/webidl-conversions": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", + "integrity": "sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==" + }, + "node_modules/whatwg-url": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz", + "integrity": "sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==", + "dependencies": { + "tr46": "~0.0.3", + "webidl-conversions": "^3.0.0" + } + }, + "node_modules/yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true + } + } +} diff --git a/package.json b/package.json new file mode 100644 index 0000000..4e5b601 --- /dev/null +++ b/package.json @@ -0,0 +1,37 @@ +{ + "name": "vite-template-solid", + "version": "0.0.0", + "description": "", + "scripts": { + "start": "vite", + "dev": "vite", + "dev:host": "vite --host", + "build": "vite build", + "build:unminified": "vite build --minify=false", + "serve": "vite preview", + "host": "vite preview --host", + "extract": "formatjs extract", + "compile": "formatjs compile" + }, + "license": "MIT", + "devDependencies": { + "@formatjs/cli": "^6.0.4", + "typescript": "^4.9.4", + "vite": "^4.0.3", + "vite-plugin-solid": "^2.5.0" + }, + "dependencies": { + "@cookbook/solid-intl": "^0.1.2", + "@jukben/emoji-search": "^2.0.1", + "@picocss/pico": "^1.5.7", + "@scure/base": "^1.1.1", + "@solidjs/router": "^0.7.0", + "@thisbeyond/solid-select": "^0.13.0", + "@types/dompurify": "^2.4.0", + "dompurify": "^3.0.0", + "link-preview-js": "^3.0.4", + "nostr-tools": "^1.4.1", + "sass": "^1.58.0", + "solid-js": "^1.6.6" + } +} diff --git a/public/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-4mF9lp.woff2 b/public/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-4mF9lp.woff2 new file mode 100644 index 0000000..210269a Binary files /dev/null and b/public/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-4mF9lp.woff2 differ diff --git a/public/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-8mF9lp.woff2 b/public/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-8mF9lp.woff2 new file mode 100644 index 0000000..5460715 Binary files /dev/null and b/public/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-8mF9lp.woff2 differ diff --git a/public/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-EmFw.woff2 b/public/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-EmFw.woff2 new file mode 100644 index 0000000..e1159ed Binary files /dev/null and b/public/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-EmFw.woff2 differ diff --git a/public/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-ImF9lp.woff2 b/public/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-ImF9lp.woff2 new file mode 100644 index 0000000..b3e3f79 Binary files /dev/null and b/public/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-ImF9lp.woff2 differ diff --git a/public/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-UmF9lp.woff2 b/public/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-UmF9lp.woff2 new file mode 100644 index 0000000..eab4010 Binary files /dev/null and b/public/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-UmF9lp.woff2 differ diff --git a/public/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-wmF9lp.woff2 b/public/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-wmF9lp.woff2 new file mode 100644 index 0000000..4f52153 Binary files /dev/null and b/public/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-wmF9lp.woff2 differ diff --git a/public/public/fonts.css b/public/public/fonts.css new file mode 100644 index 0000000..e8e8c97 --- /dev/null +++ b/public/public/fonts.css @@ -0,0 +1,180 @@ +/* cyrillic-ext */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 100; + font-stretch: 100%; + font-display: block; + src: url(/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-wmF9lp.woff2) format('woff2'); + unicode-range: U+0460-052F, U+1C80-1C88, U+20B4, U+2DE0-2DFF, U+A640-A69F, U+FE2E-FE2F; +} +/* cyrillic */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 100; + font-stretch: 100%; + font-display: block; + src: url(/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-UmF9lp.woff2) format('woff2'); + unicode-range: U+0301, U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116; +} +/* greek */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 100; + font-stretch: 100%; + font-display: block; + src: url(/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-ImF9lp.woff2) format('woff2'); + unicode-range: U+0370-03FF; +} +/* vietnamese */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 100; + font-stretch: 100%; + font-display: block; + src: url(/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-4mF9lp.woff2) format('woff2'); + unicode-range: U+0102-0103, U+0110-0111, U+0128-0129, U+0168-0169, U+01A0-01A1, U+01AF-01B0, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329, U+1EA0-1EF9, U+20AB; +} +/* latin-ext */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 100; + font-stretch: 100%; + font-display: block; + src: url(/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-8mF9lp.woff2) format('woff2'); + unicode-range: U+0100-02AF, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329, U+1E00-1EFF, U+2020, U+20A0-20AB, U+20AD-20CF, U+2113, U+2C60-2C7F, U+A720-A7FF; +} +/* latin */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 100; + font-stretch: 100%; + font-display: block; + src: url(/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-EmFw.woff2) format('woff2'); + unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD; +} +/* cyrillic-ext */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 400; + font-stretch: 100%; + font-display: block; + src: url(/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-wmF9lp.woff2) format('woff2'); + unicode-range: U+0460-052F, U+1C80-1C88, U+20B4, U+2DE0-2DFF, U+A640-A69F, U+FE2E-FE2F; +} +/* cyrillic */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 400; + font-stretch: 100%; + font-display: block; + src: url(/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-UmF9lp.woff2) format('woff2'); + unicode-range: U+0301, U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116; +} +/* greek */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 400; + font-stretch: 100%; + font-display: block; + src: url(/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-ImF9lp.woff2) format('woff2'); + unicode-range: U+0370-03FF; +} +/* vietnamese */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 400; + font-stretch: 100%; + font-display: block; + src: url(/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-4mF9lp.woff2) format('woff2'); + unicode-range: U+0102-0103, U+0110-0111, U+0128-0129, U+0168-0169, U+01A0-01A1, U+01AF-01B0, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329, U+1EA0-1EF9, U+20AB; +} +/* latin-ext */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 400; + font-stretch: 100%; + font-display: block; + src: url(/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-8mF9lp.woff2) format('woff2'); + unicode-range: U+0100-02AF, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329, U+1E00-1EFF, U+2020, U+20A0-20AB, U+20AD-20CF, U+2113, U+2C60-2C7F, U+A720-A7FF; +} +/* latin */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 400; + font-stretch: 100%; + font-display: block; + src: url(/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-EmFw.woff2) format('woff2'); + unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD; +} +/* cyrillic-ext */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 700; + font-stretch: 100%; + font-display: block; + src: url(/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-wmF9lp.woff2) format('woff2'); + unicode-range: U+0460-052F, U+1C80-1C88, U+20B4, U+2DE0-2DFF, U+A640-A69F, U+FE2E-FE2F; +} +/* cyrillic */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 700; + font-stretch: 100%; + font-display: block; + src: url(/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-UmF9lp.woff2) format('woff2'); + unicode-range: U+0301, U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116; +} +/* greek */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 700; + font-stretch: 100%; + font-display: block; + src: url(/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-ImF9lp.woff2) format('woff2'); + unicode-range: U+0370-03FF; +} +/* vietnamese */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 700; + font-stretch: 100%; + font-display: block; + src: url(/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-4mF9lp.woff2) format('woff2'); + unicode-range: U+0102-0103, U+0110-0111, U+0128-0129, U+0168-0169, U+01A0-01A1, U+01AF-01B0, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329, U+1EA0-1EF9, U+20AB; +} +/* latin-ext */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 700; + font-stretch: 100%; + font-display: block; + src: url(/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-8mF9lp.woff2) format('woff2'); + unicode-range: U+0100-02AF, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329, U+1E00-1EFF, U+2020, U+20A0-20AB, U+20AD-20CF, U+2113, U+2C60-2C7F, U+A720-A7FF; +} +/* latin */ +@font-face { + font-family: 'Roboto Flex'; + font-style: normal; + font-weight: 700; + font-stretch: 100%; + font-display: block; + src: url(/public/NaNeepOXO_NexZs0b5QrzlOHb8wCikXpYqmZsWI-__OGfttPZktqc2VdZ80KvCLZaPcSBZtOx2MifRuWR28sPJtUMbsFEK6cRrleUx9Xgbm3WLHa_F4Ep4Fm0PN19Ik5Dntczx0wZGzhPlL1YNMYKbv9_1IQXOw7AiUJVXRrV8cWW4O8LJCoXjCnwSRSaLshNP1d9-EmFw.woff2) format('woff2'); + unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+0300-0301, U+0303-0304, U+0308-0309, U+0323, U+0329, U+2000-206F, U+2074, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD; +} diff --git a/public/public/primal-thumbnail.png b/public/public/primal-thumbnail.png new file mode 100644 index 0000000..d5bcf74 Binary files /dev/null and b/public/public/primal-thumbnail.png differ diff --git a/src/App.module.scss b/src/App.module.scss new file mode 100644 index 0000000..2e9bc94 --- /dev/null +++ b/src/App.module.scss @@ -0,0 +1,6 @@ +.invisible { + position: fixed; + width: 0px !important; + height: 0px !important; + opacity: 0; +} diff --git a/src/App.tsx b/src/App.tsx new file mode 100644 index 0000000..c700e37 --- /dev/null +++ b/src/App.tsx @@ -0,0 +1,64 @@ +import { Component, onCleanup, onMount } from 'solid-js'; +import { AccountProvider } from './contexts/AccountContext'; +import { connect, disconnect } from './sockets'; +import { connect as uploadConnect, disconnect as uploadDisconnet } from './uploadSocket'; +import styles from './App.module.scss'; +import Toaster from './components/Toaster/Toaster'; +import { HomeProvider } from './contexts/HomeContext'; +import { ExploreProvider } from './contexts/ExploreContext'; +import { ThreadProvider } from './contexts/ThreadContext'; +import Router from './Router'; +import { ProfileProvider } from './contexts/ProfileContext'; +import { SettingsProvider } from './contexts/SettingsContext'; +import { TranslatorProvider } from './contexts/TranslatorContext'; +import { NotificationsProvider } from './contexts/NotificationsContext'; +import { SearchProvider } from './contexts/SearchContext'; +import { MessagesProvider } from './contexts/MessagesContext'; +import { MediaProvider } from './contexts/MediaContext'; + + +export const APP_ID = `${Math.floor(Math.random()*10000000000)}`; + +const App: Component = () => { + + onMount(() => { + connect(); + uploadConnect(); + }); + + onCleanup(() => { + disconnect(); + uploadDisconnet(); + }) + + return ( + + + + + + + + + + + + + + + + + + + + + + + + + + + ); +}; + +export default App; diff --git a/src/Router.tsx b/src/Router.tsx new file mode 100644 index 0000000..a0ac3f3 --- /dev/null +++ b/src/Router.tsx @@ -0,0 +1,91 @@ +import { Component, createReaction, createResource, lazy, Resource } from 'solid-js'; +import { Routes, Route, Navigate, RouteDataFuncArgs } from "@solidjs/router" +import Home from './pages/Home'; +import Layout from './components/Layout/Layout'; +import Explore from './pages/Explore'; +import Thread from './pages/Thread'; +import Messages from './pages/Messages'; +import Notifications from './pages/Notifications'; +import Downloads from './pages/Downloads'; +import Settings from './pages/Settings'; +import Help from './pages/Help'; +// import Profile from './pages/Profile'; +import { PrimalWindow } from './types/primal'; +import { useHomeContext } from './contexts/HomeContext'; +import { useExploreContext } from './contexts/ExploreContext'; +import { useThreadContext } from './contexts/ThreadContext'; +import { useAccountContext } from './contexts/AccountContext'; +import { useProfileContext } from './contexts/ProfileContext'; +import { useSettingsContext } from './contexts/SettingsContext'; +import NotFound from './pages/NotFound'; +import { fetchKnownProfiles } from './lib/profile'; +import Search from './pages/Search'; +import { useMessagesContext } from './contexts/MessagesContext'; +import { useMediaContext } from './contexts/MediaContext'; +import { useNotificationsContext } from './contexts/NotificationsContext'; +import { useSearchContext } from './contexts/SearchContext'; + +const primalWindow = window as PrimalWindow; + +const Router: Component = () => { + + const account = useAccountContext(); + const profile = useProfileContext(); + const settings = useSettingsContext(); + const home = useHomeContext(); + const explore = useExploreContext(); + const thread = useThreadContext(); + const messages = useMessagesContext(); + const media = useMediaContext(); + const notifications = useNotificationsContext(); + const search = useSearchContext(); + + const loadPrimalStores = () => { + primalWindow.primal = { + account, + explore, + home, + media, + messages, + notifications, + profile, + search, + settings, + thread, + }; + }; + + primalWindow.loadPrimalStores = loadPrimalStores; + + const Profile = lazy(() => import('./pages/Profile')) + + const getKnownProfiles = ({ params }: RouteDataFuncArgs) => { + const [profiles] = createResource(params.vanityName, fetchKnownProfiles) + return profiles; + } + + return ( + <> + + + } /> + + + + + + + + + + + + + + + + + ); +}; + +export default Router; diff --git a/src/assets/favicon.ico b/src/assets/favicon.ico new file mode 100644 index 0000000..d98a7c0 Binary files /dev/null and b/src/assets/favicon.ico differ diff --git a/src/assets/icons/attach_media.svg b/src/assets/icons/attach_media.svg new file mode 100644 index 0000000..4909985 --- /dev/null +++ b/src/assets/icons/attach_media.svg @@ -0,0 +1,4 @@ + + + + diff --git a/src/assets/icons/back.svg b/src/assets/icons/back.svg new file mode 100644 index 0000000..aaa6f47 --- /dev/null +++ b/src/assets/icons/back.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/caret.svg b/src/assets/icons/caret.svg new file mode 100644 index 0000000..ddb1e2f --- /dev/null +++ b/src/assets/icons/caret.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/check-black.svg b/src/assets/icons/check-black.svg new file mode 100644 index 0000000..8740937 --- /dev/null +++ b/src/assets/icons/check-black.svg @@ -0,0 +1,41 @@ + + + + + + diff --git a/src/assets/icons/check.svg b/src/assets/icons/check.svg new file mode 100644 index 0000000..67d6e1d --- /dev/null +++ b/src/assets/icons/check.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/close.svg b/src/assets/icons/close.svg new file mode 100644 index 0000000..39660b3 --- /dev/null +++ b/src/assets/icons/close.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/context.svg b/src/assets/icons/context.svg new file mode 100644 index 0000000..b7c0e51 --- /dev/null +++ b/src/assets/icons/context.svg @@ -0,0 +1,5 @@ + + + + + diff --git a/src/assets/icons/copy.svg b/src/assets/icons/copy.svg new file mode 100644 index 0000000..6b579e0 --- /dev/null +++ b/src/assets/icons/copy.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/corner_left.svg b/src/assets/icons/corner_left.svg new file mode 100644 index 0000000..8b09421 --- /dev/null +++ b/src/assets/icons/corner_left.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/corner_right.svg b/src/assets/icons/corner_right.svg new file mode 100644 index 0000000..e6746a9 --- /dev/null +++ b/src/assets/icons/corner_right.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/default_avatar.svg b/src/assets/icons/default_avatar.svg new file mode 100644 index 0000000..d6dffe6 --- /dev/null +++ b/src/assets/icons/default_avatar.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/default_nostrich.svg b/src/assets/icons/default_nostrich.svg new file mode 100644 index 0000000..521fbbe --- /dev/null +++ b/src/assets/icons/default_nostrich.svg @@ -0,0 +1,10 @@ + + + + + + + + + + diff --git a/src/assets/icons/delete.svg b/src/assets/icons/delete.svg new file mode 100644 index 0000000..d15a3f4 --- /dev/null +++ b/src/assets/icons/delete.svg @@ -0,0 +1,6 @@ + + + + + + diff --git a/src/assets/icons/download.svg b/src/assets/icons/download.svg new file mode 100644 index 0000000..87d8a81 --- /dev/null +++ b/src/assets/icons/download.svg @@ -0,0 +1,4 @@ + + + + diff --git a/src/assets/icons/drag_handle.svg b/src/assets/icons/drag_handle.svg new file mode 100644 index 0000000..8eea3e9 --- /dev/null +++ b/src/assets/icons/drag_handle.svg @@ -0,0 +1,10 @@ + + + + + + + + + + diff --git a/src/assets/icons/edit.svg b/src/assets/icons/edit.svg new file mode 100644 index 0000000..ccaa264 --- /dev/null +++ b/src/assets/icons/edit.svg @@ -0,0 +1,4 @@ + + + + diff --git a/src/assets/icons/explore.svg b/src/assets/icons/explore.svg new file mode 100644 index 0000000..4472d74 --- /dev/null +++ b/src/assets/icons/explore.svg @@ -0,0 +1,5 @@ + + + + + diff --git a/src/assets/icons/explore/clock.svg b/src/assets/icons/explore/clock.svg new file mode 100644 index 0000000..a83a8e8 --- /dev/null +++ b/src/assets/icons/explore/clock.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/explore/flame.svg b/src/assets/icons/explore/flame.svg new file mode 100644 index 0000000..48f1b12 --- /dev/null +++ b/src/assets/icons/explore/flame.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/explore/follows.svg b/src/assets/icons/explore/follows.svg new file mode 100644 index 0000000..8e5be78 --- /dev/null +++ b/src/assets/icons/explore/follows.svg @@ -0,0 +1,8 @@ + + + + + + + + diff --git a/src/assets/icons/explore/global.svg b/src/assets/icons/explore/global.svg new file mode 100644 index 0000000..4ad5abf --- /dev/null +++ b/src/assets/icons/explore/global.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/explore/likes.svg b/src/assets/icons/explore/likes.svg new file mode 100644 index 0000000..f78c6f6 --- /dev/null +++ b/src/assets/icons/explore/likes.svg @@ -0,0 +1,4 @@ + + + + diff --git a/src/assets/icons/explore/network.svg b/src/assets/icons/explore/network.svg new file mode 100644 index 0000000..552fe89 --- /dev/null +++ b/src/assets/icons/explore/network.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/explore/tribe.svg b/src/assets/icons/explore/tribe.svg new file mode 100644 index 0000000..19328bb --- /dev/null +++ b/src/assets/icons/explore/tribe.svg @@ -0,0 +1,16 @@ + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/explore/zaps.svg b/src/assets/icons/explore/zaps.svg new file mode 100644 index 0000000..a081611 --- /dev/null +++ b/src/assets/icons/explore/zaps.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/explore/zaps_hollow.svg b/src/assets/icons/explore/zaps_hollow.svg new file mode 100644 index 0000000..be8c07b --- /dev/null +++ b/src/assets/icons/explore/zaps_hollow.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/feed_add.svg b/src/assets/icons/feed_add.svg new file mode 100644 index 0000000..88a50c7 --- /dev/null +++ b/src/assets/icons/feed_add.svg @@ -0,0 +1,6 @@ + + + + + + diff --git a/src/assets/icons/feed_like.svg b/src/assets/icons/feed_like.svg new file mode 100644 index 0000000..bd34559 --- /dev/null +++ b/src/assets/icons/feed_like.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/feed_like_fill.svg b/src/assets/icons/feed_like_fill.svg new file mode 100644 index 0000000..ff5ad67 --- /dev/null +++ b/src/assets/icons/feed_like_fill.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/feed_picker.svg b/src/assets/icons/feed_picker.svg new file mode 100644 index 0000000..259fd56 --- /dev/null +++ b/src/assets/icons/feed_picker.svg @@ -0,0 +1,5 @@ + + + + + diff --git a/src/assets/icons/feed_remove.svg b/src/assets/icons/feed_remove.svg new file mode 100644 index 0000000..e588f73 --- /dev/null +++ b/src/assets/icons/feed_remove.svg @@ -0,0 +1,6 @@ + + + + + + diff --git a/src/assets/icons/feed_reply.svg b/src/assets/icons/feed_reply.svg new file mode 100644 index 0000000..2f254a0 --- /dev/null +++ b/src/assets/icons/feed_reply.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/feed_reply_fill.svg b/src/assets/icons/feed_reply_fill.svg new file mode 100644 index 0000000..2964b56 --- /dev/null +++ b/src/assets/icons/feed_reply_fill.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/feed_repost.svg b/src/assets/icons/feed_repost.svg new file mode 100644 index 0000000..106e48d --- /dev/null +++ b/src/assets/icons/feed_repost.svg @@ -0,0 +1,4 @@ + + + + diff --git a/src/assets/icons/feed_repost_fill.svg b/src/assets/icons/feed_repost_fill.svg new file mode 100644 index 0000000..93467cb --- /dev/null +++ b/src/assets/icons/feed_repost_fill.svg @@ -0,0 +1,4 @@ + + + + diff --git a/src/assets/icons/feed_zap.svg b/src/assets/icons/feed_zap.svg new file mode 100644 index 0000000..472277d --- /dev/null +++ b/src/assets/icons/feed_zap.svg @@ -0,0 +1,10 @@ + + + + + + + + + + diff --git a/src/assets/icons/feed_zap_fill.svg b/src/assets/icons/feed_zap_fill.svg new file mode 100644 index 0000000..dbf907e --- /dev/null +++ b/src/assets/icons/feed_zap_fill.svg @@ -0,0 +1,10 @@ + + + + + + + + + + diff --git a/src/assets/icons/follows.svg b/src/assets/icons/follows.svg new file mode 100644 index 0000000..3ea8590 --- /dev/null +++ b/src/assets/icons/follows.svg @@ -0,0 +1,34 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/follows_latest.svg b/src/assets/icons/follows_latest.svg new file mode 100644 index 0000000..cd65369 --- /dev/null +++ b/src/assets/icons/follows_latest.svg @@ -0,0 +1,46 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/follows_latest_light.svg b/src/assets/icons/follows_latest_light.svg new file mode 100644 index 0000000..1c3298d --- /dev/null +++ b/src/assets/icons/follows_latest_light.svg @@ -0,0 +1,52 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/follows_light.svg b/src/assets/icons/follows_light.svg new file mode 100644 index 0000000..a31768e --- /dev/null +++ b/src/assets/icons/follows_light.svg @@ -0,0 +1,134 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/follows_popular.svg b/src/assets/icons/follows_popular.svg new file mode 100644 index 0000000..a392ac5 --- /dev/null +++ b/src/assets/icons/follows_popular.svg @@ -0,0 +1,47 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/follows_popular_light.svg b/src/assets/icons/follows_popular_light.svg new file mode 100644 index 0000000..6d51af2 --- /dev/null +++ b/src/assets/icons/follows_popular_light.svg @@ -0,0 +1,47 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/follows_trending.svg b/src/assets/icons/follows_trending.svg new file mode 100644 index 0000000..e5f3c37 --- /dev/null +++ b/src/assets/icons/follows_trending.svg @@ -0,0 +1,47 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/follows_trending_light.svg b/src/assets/icons/follows_trending_light.svg new file mode 100644 index 0000000..2857de7 --- /dev/null +++ b/src/assets/icons/follows_trending_light.svg @@ -0,0 +1,47 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/forward.svg b/src/assets/icons/forward.svg new file mode 100644 index 0000000..89ef8db --- /dev/null +++ b/src/assets/icons/forward.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/get_started.svg b/src/assets/icons/get_started.svg new file mode 100644 index 0000000..c1837d0 --- /dev/null +++ b/src/assets/icons/get_started.svg @@ -0,0 +1,15 @@ + + + + + + + + + + + + + + + diff --git a/src/assets/icons/github.svg b/src/assets/icons/github.svg new file mode 100644 index 0000000..0223dcb --- /dev/null +++ b/src/assets/icons/github.svg @@ -0,0 +1,10 @@ + + + + + + + + + + diff --git a/src/assets/icons/github_light.svg b/src/assets/icons/github_light.svg new file mode 100644 index 0000000..4669af8 --- /dev/null +++ b/src/assets/icons/github_light.svg @@ -0,0 +1,10 @@ + + + + + + + + + + diff --git a/src/assets/icons/global.svg b/src/assets/icons/global.svg new file mode 100644 index 0000000..7bdfc32 --- /dev/null +++ b/src/assets/icons/global.svg @@ -0,0 +1,10 @@ + + + + + + + + + + diff --git a/src/assets/icons/global_latest.svg b/src/assets/icons/global_latest.svg new file mode 100644 index 0000000..43d050e --- /dev/null +++ b/src/assets/icons/global_latest.svg @@ -0,0 +1,21 @@ + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/global_latest_light.svg b/src/assets/icons/global_latest_light.svg new file mode 100644 index 0000000..5aec3ea --- /dev/null +++ b/src/assets/icons/global_latest_light.svg @@ -0,0 +1,27 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/global_light.svg b/src/assets/icons/global_light.svg new file mode 100644 index 0000000..b806c34 --- /dev/null +++ b/src/assets/icons/global_light.svg @@ -0,0 +1,9 @@ + + + + + + + + + diff --git a/src/assets/icons/global_popular.svg b/src/assets/icons/global_popular.svg new file mode 100644 index 0000000..990407b --- /dev/null +++ b/src/assets/icons/global_popular.svg @@ -0,0 +1,29 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/global_popular_light.svg b/src/assets/icons/global_popular_light.svg new file mode 100644 index 0000000..b1ab616 --- /dev/null +++ b/src/assets/icons/global_popular_light.svg @@ -0,0 +1,22 @@ + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/global_trending.svg b/src/assets/icons/global_trending.svg new file mode 100644 index 0000000..04a40fe --- /dev/null +++ b/src/assets/icons/global_trending.svg @@ -0,0 +1,22 @@ + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/global_trending_light.svg b/src/assets/icons/global_trending_light.svg new file mode 100644 index 0000000..734fa7e --- /dev/null +++ b/src/assets/icons/global_trending_light.svg @@ -0,0 +1,22 @@ + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/golbal_popular.svg b/src/assets/icons/golbal_popular.svg new file mode 100644 index 0000000..d00562f --- /dev/null +++ b/src/assets/icons/golbal_popular.svg @@ -0,0 +1,22 @@ + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/help.svg b/src/assets/icons/help.svg new file mode 100644 index 0000000..d624030 --- /dev/null +++ b/src/assets/icons/help.svg @@ -0,0 +1,5 @@ + + + + + diff --git a/src/assets/icons/home.svg b/src/assets/icons/home.svg new file mode 100644 index 0000000..5843407 --- /dev/null +++ b/src/assets/icons/home.svg @@ -0,0 +1,4 @@ + + + + diff --git a/src/assets/icons/key.svg b/src/assets/icons/key.svg new file mode 100644 index 0000000..de44eba --- /dev/null +++ b/src/assets/icons/key.svg @@ -0,0 +1,10 @@ + + + + + + + + + + diff --git a/src/assets/icons/like.svg b/src/assets/icons/like.svg new file mode 100644 index 0000000..d3a038c --- /dev/null +++ b/src/assets/icons/like.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/like_filled.svg b/src/assets/icons/like_filled.svg new file mode 100644 index 0000000..58c9777 --- /dev/null +++ b/src/assets/icons/like_filled.svg @@ -0,0 +1,4 @@ + + + + diff --git a/src/assets/icons/link.svg b/src/assets/icons/link.svg new file mode 100644 index 0000000..1f7d157 --- /dev/null +++ b/src/assets/icons/link.svg @@ -0,0 +1,4 @@ + + + + diff --git a/src/assets/icons/logo.svg b/src/assets/icons/logo.svg new file mode 100644 index 0000000..d43e531 --- /dev/null +++ b/src/assets/icons/logo.svg @@ -0,0 +1,24 @@ + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/logo_fire.svg b/src/assets/icons/logo_fire.svg new file mode 100644 index 0000000..d584495 --- /dev/null +++ b/src/assets/icons/logo_fire.svg @@ -0,0 +1,24 @@ + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/logo_ice.svg b/src/assets/icons/logo_ice.svg new file mode 100644 index 0000000..04dcdcd --- /dev/null +++ b/src/assets/icons/logo_ice.svg @@ -0,0 +1,29 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/messages.svg b/src/assets/icons/messages.svg new file mode 100644 index 0000000..113406f --- /dev/null +++ b/src/assets/icons/messages.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/miljan.jpg b/src/assets/icons/miljan.jpg new file mode 100644 index 0000000..2e088d6 Binary files /dev/null and b/src/assets/icons/miljan.jpg differ diff --git a/src/assets/icons/network.svg b/src/assets/icons/network.svg new file mode 100644 index 0000000..fe9e268 --- /dev/null +++ b/src/assets/icons/network.svg @@ -0,0 +1,10 @@ + + + + + + + + + + diff --git a/src/assets/icons/network_latest.svg b/src/assets/icons/network_latest.svg new file mode 100644 index 0000000..aebb78d --- /dev/null +++ b/src/assets/icons/network_latest.svg @@ -0,0 +1,21 @@ + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/network_latest_light.svg b/src/assets/icons/network_latest_light.svg new file mode 100644 index 0000000..d5e9566 --- /dev/null +++ b/src/assets/icons/network_latest_light.svg @@ -0,0 +1,27 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/network_light.svg b/src/assets/icons/network_light.svg new file mode 100644 index 0000000..b931d10 --- /dev/null +++ b/src/assets/icons/network_light.svg @@ -0,0 +1,9 @@ + + + + + + + + + diff --git a/src/assets/icons/network_popular.svg b/src/assets/icons/network_popular.svg new file mode 100644 index 0000000..c1d8ebd --- /dev/null +++ b/src/assets/icons/network_popular.svg @@ -0,0 +1,22 @@ + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/network_popular_light.svg b/src/assets/icons/network_popular_light.svg new file mode 100644 index 0000000..a2ca1d3 --- /dev/null +++ b/src/assets/icons/network_popular_light.svg @@ -0,0 +1,22 @@ + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/network_trending.svg b/src/assets/icons/network_trending.svg new file mode 100644 index 0000000..8842ef5 --- /dev/null +++ b/src/assets/icons/network_trending.svg @@ -0,0 +1,22 @@ + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/network_trending_light.svg b/src/assets/icons/network_trending_light.svg new file mode 100644 index 0000000..b37ac28 --- /dev/null +++ b/src/assets/icons/network_trending_light.svg @@ -0,0 +1,22 @@ + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/notifications.svg b/src/assets/icons/notifications.svg new file mode 100644 index 0000000..8812646 --- /dev/null +++ b/src/assets/icons/notifications.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/notifications/at.svg b/src/assets/icons/notifications/at.svg new file mode 100644 index 0000000..3baca47 --- /dev/null +++ b/src/assets/icons/notifications/at.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/notifications/follows.svg b/src/assets/icons/notifications/follows.svg new file mode 100644 index 0000000..921d9c4 --- /dev/null +++ b/src/assets/icons/notifications/follows.svg @@ -0,0 +1,6 @@ + + + + + + diff --git a/src/assets/icons/notifications/mention.svg b/src/assets/icons/notifications/mention.svg new file mode 100644 index 0000000..ac59246 --- /dev/null +++ b/src/assets/icons/notifications/mention.svg @@ -0,0 +1,6 @@ + + + + + + diff --git a/src/assets/icons/notifications/mention_liked.svg b/src/assets/icons/notifications/mention_liked.svg new file mode 100644 index 0000000..e48e710 --- /dev/null +++ b/src/assets/icons/notifications/mention_liked.svg @@ -0,0 +1,5 @@ + + + + + diff --git a/src/assets/icons/notifications/mention_replied.svg b/src/assets/icons/notifications/mention_replied.svg new file mode 100644 index 0000000..89bb4ae --- /dev/null +++ b/src/assets/icons/notifications/mention_replied.svg @@ -0,0 +1,5 @@ + + + + + diff --git a/src/assets/icons/notifications/mention_reposted.svg b/src/assets/icons/notifications/mention_reposted.svg new file mode 100644 index 0000000..7ba4683 --- /dev/null +++ b/src/assets/icons/notifications/mention_reposted.svg @@ -0,0 +1,5 @@ + + + + + diff --git a/src/assets/icons/notifications/mention_zapped.svg b/src/assets/icons/notifications/mention_zapped.svg new file mode 100644 index 0000000..30d51be --- /dev/null +++ b/src/assets/icons/notifications/mention_zapped.svg @@ -0,0 +1,5 @@ + + + + + diff --git a/src/assets/icons/notifications/mentioned_post.svg b/src/assets/icons/notifications/mentioned_post.svg new file mode 100644 index 0000000..14f0397 --- /dev/null +++ b/src/assets/icons/notifications/mentioned_post.svg @@ -0,0 +1,5 @@ + + + + + diff --git a/src/assets/icons/notifications/mentioned_post_liked.svg b/src/assets/icons/notifications/mentioned_post_liked.svg new file mode 100644 index 0000000..3a049cf --- /dev/null +++ b/src/assets/icons/notifications/mentioned_post_liked.svg @@ -0,0 +1,4 @@ + + + + diff --git a/src/assets/icons/notifications/mentioned_post_replied.svg b/src/assets/icons/notifications/mentioned_post_replied.svg new file mode 100644 index 0000000..6614cac --- /dev/null +++ b/src/assets/icons/notifications/mentioned_post_replied.svg @@ -0,0 +1,4 @@ + + + + diff --git a/src/assets/icons/notifications/mentioned_post_reposted.svg b/src/assets/icons/notifications/mentioned_post_reposted.svg new file mode 100644 index 0000000..c5a52aa --- /dev/null +++ b/src/assets/icons/notifications/mentioned_post_reposted.svg @@ -0,0 +1,4 @@ + + + + diff --git a/src/assets/icons/notifications/mentioned_post_zapped.svg b/src/assets/icons/notifications/mentioned_post_zapped.svg new file mode 100644 index 0000000..f48f526 --- /dev/null +++ b/src/assets/icons/notifications/mentioned_post_zapped.svg @@ -0,0 +1,4 @@ + + + + diff --git a/src/assets/icons/notifications/post_liked.svg b/src/assets/icons/notifications/post_liked.svg new file mode 100644 index 0000000..b2a1f57 --- /dev/null +++ b/src/assets/icons/notifications/post_liked.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/notifications/post_replied.svg b/src/assets/icons/notifications/post_replied.svg new file mode 100644 index 0000000..e1badbe --- /dev/null +++ b/src/assets/icons/notifications/post_replied.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/notifications/post_reposted.svg b/src/assets/icons/notifications/post_reposted.svg new file mode 100644 index 0000000..3244ed9 --- /dev/null +++ b/src/assets/icons/notifications/post_reposted.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/notifications/post_zapped.svg b/src/assets/icons/notifications/post_zapped.svg new file mode 100644 index 0000000..ff69da8 --- /dev/null +++ b/src/assets/icons/notifications/post_zapped.svg @@ -0,0 +1,13 @@ + + + + + + + + + + + + + diff --git a/src/assets/icons/notifications/user_followed.svg b/src/assets/icons/notifications/user_followed.svg new file mode 100644 index 0000000..802cb5f --- /dev/null +++ b/src/assets/icons/notifications/user_followed.svg @@ -0,0 +1,7 @@ + + + + + + + diff --git a/src/assets/icons/notifications/user_unfollowed.svg b/src/assets/icons/notifications/user_unfollowed.svg new file mode 100644 index 0000000..8dc46a2 --- /dev/null +++ b/src/assets/icons/notifications/user_unfollowed.svg @@ -0,0 +1,7 @@ + + + + + + + diff --git a/src/assets/icons/old/download_filled.svg b/src/assets/icons/old/download_filled.svg new file mode 100644 index 0000000..839e888 --- /dev/null +++ b/src/assets/icons/old/download_filled.svg @@ -0,0 +1,4 @@ + + + + diff --git a/src/assets/icons/old/downloads.svg b/src/assets/icons/old/downloads.svg new file mode 100644 index 0000000..7acd348 --- /dev/null +++ b/src/assets/icons/old/downloads.svg @@ -0,0 +1,6 @@ + + + + + + diff --git a/src/assets/icons/old/explore.svg b/src/assets/icons/old/explore.svg new file mode 100644 index 0000000..a535d40 --- /dev/null +++ b/src/assets/icons/old/explore.svg @@ -0,0 +1,6 @@ + + + + + + diff --git a/src/assets/icons/old/explore_filled.svg b/src/assets/icons/old/explore_filled.svg new file mode 100644 index 0000000..41daf35 --- /dev/null +++ b/src/assets/icons/old/explore_filled.svg @@ -0,0 +1,11 @@ + + + + + + + + + + + diff --git a/src/assets/icons/old/help.svg b/src/assets/icons/old/help.svg new file mode 100644 index 0000000..8e8620d --- /dev/null +++ b/src/assets/icons/old/help.svg @@ -0,0 +1,5 @@ + + + + + diff --git a/src/assets/icons/old/help_filled.svg b/src/assets/icons/old/help_filled.svg new file mode 100644 index 0000000..2d7d9dd --- /dev/null +++ b/src/assets/icons/old/help_filled.svg @@ -0,0 +1,10 @@ + + + + + + + + + + diff --git a/src/assets/icons/old/home-only.svg b/src/assets/icons/old/home-only.svg new file mode 100644 index 0000000..204b650 --- /dev/null +++ b/src/assets/icons/old/home-only.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/old/home.svg b/src/assets/icons/old/home.svg new file mode 100644 index 0000000..74adf86 --- /dev/null +++ b/src/assets/icons/old/home.svg @@ -0,0 +1,14 @@ + + + + + + + + + + + + + + diff --git a/src/assets/icons/old/home_filled.svg b/src/assets/icons/old/home_filled.svg new file mode 100644 index 0000000..29e2d3c --- /dev/null +++ b/src/assets/icons/old/home_filled.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/old/messages.svg b/src/assets/icons/old/messages.svg new file mode 100644 index 0000000..1f71691 --- /dev/null +++ b/src/assets/icons/old/messages.svg @@ -0,0 +1,10 @@ + + + + + + + + + + diff --git a/src/assets/icons/old/messages_filled.svg b/src/assets/icons/old/messages_filled.svg new file mode 100644 index 0000000..e842aab --- /dev/null +++ b/src/assets/icons/old/messages_filled.svg @@ -0,0 +1,10 @@ + + + + + + + + + + diff --git a/src/assets/icons/old/notifications.svg b/src/assets/icons/old/notifications.svg new file mode 100644 index 0000000..c9798a0 --- /dev/null +++ b/src/assets/icons/old/notifications.svg @@ -0,0 +1,11 @@ + + + + + + + + + + + diff --git a/src/assets/icons/old/notifications_filled.svg b/src/assets/icons/old/notifications_filled.svg new file mode 100644 index 0000000..c64db85 --- /dev/null +++ b/src/assets/icons/old/notifications_filled.svg @@ -0,0 +1,4 @@ + + + + diff --git a/src/assets/icons/old/settings.svg b/src/assets/icons/old/settings.svg new file mode 100644 index 0000000..418b7f1 --- /dev/null +++ b/src/assets/icons/old/settings.svg @@ -0,0 +1,4 @@ + + + + diff --git a/src/assets/icons/old/settings_filled.svg b/src/assets/icons/old/settings_filled.svg new file mode 100644 index 0000000..78d2ca1 --- /dev/null +++ b/src/assets/icons/old/settings_filled.svg @@ -0,0 +1,10 @@ + + + + + + + + + + diff --git a/src/assets/icons/post.svg b/src/assets/icons/post.svg new file mode 100644 index 0000000..dd75152 --- /dev/null +++ b/src/assets/icons/post.svg @@ -0,0 +1,4 @@ + + + + diff --git a/src/assets/icons/reply.svg b/src/assets/icons/reply.svg new file mode 100644 index 0000000..3179623 --- /dev/null +++ b/src/assets/icons/reply.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/reply_filled.svg b/src/assets/icons/reply_filled.svg new file mode 100644 index 0000000..b34b012 --- /dev/null +++ b/src/assets/icons/reply_filled.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/reposts.svg b/src/assets/icons/reposts.svg new file mode 100644 index 0000000..eee660d --- /dev/null +++ b/src/assets/icons/reposts.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/search.svg b/src/assets/icons/search.svg new file mode 100644 index 0000000..8390f42 --- /dev/null +++ b/src/assets/icons/search.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/search_filled.svg b/src/assets/icons/search_filled.svg new file mode 100644 index 0000000..2df437e --- /dev/null +++ b/src/assets/icons/search_filled.svg @@ -0,0 +1,4 @@ + + + + diff --git a/src/assets/icons/settings.svg b/src/assets/icons/settings.svg new file mode 100644 index 0000000..ae9a0f0 --- /dev/null +++ b/src/assets/icons/settings.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/tribe.svg b/src/assets/icons/tribe.svg new file mode 100644 index 0000000..06731e7 --- /dev/null +++ b/src/assets/icons/tribe.svg @@ -0,0 +1,74 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/tribe_latest.svg b/src/assets/icons/tribe_latest.svg new file mode 100644 index 0000000..bdb499f --- /dev/null +++ b/src/assets/icons/tribe_latest.svg @@ -0,0 +1,71 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/tribe_latest_light.svg b/src/assets/icons/tribe_latest_light.svg new file mode 100644 index 0000000..05a8360 --- /dev/null +++ b/src/assets/icons/tribe_latest_light.svg @@ -0,0 +1,77 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/tribe_light.svg b/src/assets/icons/tribe_light.svg new file mode 100644 index 0000000..cbe8e2e --- /dev/null +++ b/src/assets/icons/tribe_light.svg @@ -0,0 +1,144 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/tribe_popular.svg b/src/assets/icons/tribe_popular.svg new file mode 100644 index 0000000..0416909 --- /dev/null +++ b/src/assets/icons/tribe_popular.svg @@ -0,0 +1,72 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/tribe_popular_light.svg b/src/assets/icons/tribe_popular_light.svg new file mode 100644 index 0000000..d72b333 --- /dev/null +++ b/src/assets/icons/tribe_popular_light.svg @@ -0,0 +1,72 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/tribe_trending.svg b/src/assets/icons/tribe_trending.svg new file mode 100644 index 0000000..2d11698 --- /dev/null +++ b/src/assets/icons/tribe_trending.svg @@ -0,0 +1,72 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/tribe_trending_light.svg b/src/assets/icons/tribe_trending_light.svg new file mode 100644 index 0000000..aed8840 --- /dev/null +++ b/src/assets/icons/tribe_trending_light.svg @@ -0,0 +1,72 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/assets/icons/verified.svg b/src/assets/icons/verified.svg new file mode 100644 index 0000000..4a94d12 --- /dev/null +++ b/src/assets/icons/verified.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/zaps.svg b/src/assets/icons/zaps.svg new file mode 100644 index 0000000..bbcb0ad --- /dev/null +++ b/src/assets/icons/zaps.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/icons/zaps_filled.svg b/src/assets/icons/zaps_filled.svg new file mode 100644 index 0000000..7fa2904 --- /dev/null +++ b/src/assets/icons/zaps_filled.svg @@ -0,0 +1,3 @@ + + + diff --git a/src/assets/lottie/like.json b/src/assets/lottie/like.json new file mode 100644 index 0000000..6f816c7 --- /dev/null +++ b/src/assets/lottie/like.json @@ -0,0 +1 @@ +{"v":"4.8.0","meta":{"g":"LottieFiles AE 3.4.3","a":"","k":"","d":"","tc":""},"fr":30,"ip":0,"op":44,"w":1600,"h":1600,"nm":"Icon - Like","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":1,"ty":4,"nm":"Like Color 2","parent":5,"td":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":7,"s":[-2,434.828,0],"to":[0.333,-72.471,0],"ti":[-0.333,72.471,0]},{"t":14,"s":[0,0,0]}],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.4,0.4,0.4],"y":[1,1,1]},"o":{"x":[0.6,0.6,0.6],"y":[0,0,0]},"t":7,"s":[36,36,100]},{"t":14,"s":[100,100,100]}],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.5,"y":1},"o":{"x":0.19,"y":0},"t":0,"s":[{"i":[[0,0],[0,0],[0.934,0.862],[3.208,3.137],[9.951,11.279],[19.234,37.944],[-4.372,48.876],[-50.933,43.057],[-48.268,-3.073],[-30.424,-24.711],[-15.223,-20.401],[-25.28,20.533],[-47.222,3.006],[-50.09,-42.34],[-10.435,-44.105],[13.083,-33.182],[15.403,-18.231],[5.198,-5.073],[1.581,-1.425],[0.436,-0.383],[0,0],[9.529,9.397]],"o":[[0,0],[-0.542,-0.487],[-1.866,-1.722],[-6.406,-6.266],[-19.739,-22.371],[-19.146,-37.773],[4.485,-50.144],[50.086,-42.341],[47.221,3.006],[25.28,20.534],[15.222,-20.401],[30.424,-24.711],[48.264,-3.073],[48.974,41.395],[10.396,43.936],[-12.985,32.927],[-7.782,9.213],[-2.604,2.542],[-0.622,0.563],[0,0],[-9.529,9.397],[0,0]],"v":[[-284.573,97.974],[-284.622,97.93],[-286.845,95.904],[-294.535,88.584],[-319.71,62.021],[-383.67,-30.448],[-413.106,-163.021],[-333.514,-305.328],[-184.822,-361.596],[-67.552,-312.007],[-6.532,-247.313],[54.486,-312.006],[171.757,-361.596],[320.45,-305.329],[407.082,-175.984],[397.234,-58.463],[349.699,19.47],[329.671,41.044],[323.328,47.014],[321.737,48.434],[10.655,355.271],[-23.716,355.271]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":5,"s":[{"i":[[15.537,10.805],[0,0],[0,0],[0,0],[0,0],[24.175,34.947],[-4.877,38.506],[-56.821,33.922],[-53.848,-2.421],[-33.941,-19.469],[-16.983,-16.073],[-28.203,16.177],[-52.681,2.369],[-55.881,-33.356],[-11.642,-34.748],[13.993,-26.374],[0,0],[0,0],[1.764,-1.123],[0,0],[204.541,-93.271],[15.378,6.376]],"o":[[-15.537,-10.805],[-0.604,-0.383],[0,0],[0,0],[0,0],[-20.785,-30.046],[5.004,-39.505],[55.876,-33.358],[52.68,2.368],[28.203,16.177],[16.981,-16.073],[33.941,-19.469],[53.844,-2.421],[54.635,32.613],[11.597,34.614],[-31.498,59.367],[0,0],[0,0],[-0.694,0.444],[0,0],[-11.329,3.018],[-202.799,-88.104]],"v":[[-303.824,211.323],[-330.653,190.152],[-350.982,173.525],[-368.486,159.303],[-397.687,131.331],[-435.015,80.555],[-465.623,-46.906],[-371.81,-163.249],[-205.928,-207.579],[-75.101,-168.51],[-7.027,-117.542],[61.045,-168.51],[191.873,-207.579],[357.756,-163.25],[454.403,-61.346],[449.552,41.575],[348.551,159.808],[313.936,186.669],[294.309,201.001],[271.896,217.15],[12.146,357.199],[-26.197,357.199]],"c":true}]},{"i":{"x":0.833,"y":1},"o":{"x":0.167,"y":0},"t":12,"s":[{"i":[[0,0],[0,0],[0.819,0.913],[2.814,3.325],[8.731,11.953],[16.875,40.213],[-3.836,51.798],[-44.687,45.632],[-42.348,-3.257],[-26.693,-26.189],[-13.356,-21.621],[-22.18,21.761],[-41.431,3.186],[-43.948,-44.871],[-9.156,-46.742],[11.481,-35.165],[13.512,-19.323],[4.56,-5.377],[1.387,-1.51],[0.382,-0.406],[0,0],[8.36,9.959]],"o":[[0,0],[-0.475,-0.516],[-1.637,-1.825],[-5.62,-6.641],[-17.318,-23.709],[-16.798,-40.031],[3.935,-53.142],[43.943,-44.873],[41.43,3.186],[22.18,21.761],[13.355,-21.621],[26.693,-26.189],[42.345,-3.257],[42.968,43.87],[9.121,46.563],[-11.393,34.896],[-6.828,9.764],[-2.285,2.694],[-0.545,0.597],[0,0],[-8.361,9.959],[0,0]],"v":[[-249.674,103.832],[-249.717,103.786],[-251.667,101.638],[-258.414,93.88],[-280.502,65.729],[-336.618,-32.269],[-362.443,-172.769],[-292.613,-323.585],[-162.156,-383.217],[-59.268,-330.662],[-5.731,-262.1],[47.804,-330.662],[150.693,-383.217],[281.151,-323.586],[357.159,-186.507],[348.519,-61.958],[306.813,20.634],[289.241,43.499],[283.676,49.825],[282.28,51.33],[9.348,376.514],[-20.807,376.514]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.167,"y":0},"t":17,"s":[{"i":[[0,0],[0,0],[0.934,0.862],[3.208,3.137],[9.951,11.279],[19.234,37.944],[-4.372,48.876],[-50.933,43.057],[-48.268,-3.073],[-30.424,-24.711],[-15.223,-20.401],[-25.28,20.533],[-47.222,3.006],[-50.09,-42.34],[-10.435,-44.105],[13.083,-33.182],[15.403,-18.231],[5.198,-5.073],[1.581,-1.425],[0.436,-0.383],[0,0],[9.529,9.397]],"o":[[0,0],[-0.542,-0.487],[-1.866,-1.722],[-6.406,-6.266],[-19.739,-22.371],[-19.146,-37.773],[4.485,-50.144],[50.086,-42.341],[47.221,3.006],[25.28,20.534],[15.222,-20.401],[30.424,-24.711],[48.264,-3.073],[48.974,41.395],[10.396,43.936],[-12.985,32.927],[-7.782,9.213],[-2.604,2.542],[-0.622,0.563],[0,0],[-9.529,9.397],[0,0]],"v":[[-284.573,97.974],[-284.622,97.93],[-286.845,95.904],[-294.535,88.584],[-319.71,62.021],[-383.67,-30.448],[-413.106,-163.021],[-333.514,-305.328],[-184.822,-361.596],[-67.552,-312.007],[-6.532,-247.313],[54.486,-312.006],[171.757,-361.596],[320.45,-305.329],[407.082,-175.984],[397.234,-58.463],[349.699,19.47],[329.671,41.044],[323.328,47.014],[321.737,48.434],[10.655,355.271],[-23.716,355.271]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":20,"s":[{"i":[[14.574,11.501],[0,0],[0,0],[0,0],[0,0],[22.678,37.202],[-4.575,40.99],[-53.302,36.11],[-50.513,-2.577],[-31.839,-20.725],[-15.931,-17.11],[-26.456,17.221],[-49.418,2.521],[-52.421,-35.508],[-10.921,-36.989],[13.127,-28.076],[0,0],[0,0],[1.654,-1.195],[0,0],[191.873,-99.288],[14.425,6.788]],"o":[[-14.574,-11.502],[-0.567,-0.408],[0,0],[0,0],[0,0],[-19.497,-31.985],[4.694,-42.054],[52.415,-35.51],[49.418,2.521],[26.456,17.221],[15.93,-17.11],[31.839,-20.725],[50.509,-2.577],[51.252,34.716],[10.879,36.847],[-29.547,63.196],[0,0],[0,0],[-0.651,0.473],[0,0],[-10.627,3.212],[-190.239,-93.788]],"v":[[-285.147,201.501],[-310.314,178.965],[-329.384,161.265],[-345.804,146.126],[-373.197,116.349],[-408.213,62.298],[-436.925,-73.386],[-348.922,-197.233],[-193.314,-244.423],[-70.589,-202.834],[-6.731,-148.578],[57.125,-202.834],[179.85,-244.423],[335.46,-197.234],[426.121,-88.757],[421.571,20.804],[326.825,146.664],[294.354,175.257],[275.942,190.513],[254.917,207.705],[21.255,356.575],[-42.714,356.575]],"c":true}]},{"t":24,"s":[{"i":[[0,0],[0,0],[0.934,0.862],[3.208,3.137],[9.951,11.279],[19.234,37.944],[-4.372,48.876],[-50.933,43.057],[-48.268,-3.073],[-30.424,-24.711],[-15.223,-20.401],[-25.28,20.533],[-47.222,3.006],[-50.09,-42.34],[-10.435,-44.105],[13.083,-33.182],[15.403,-18.231],[5.198,-5.073],[1.581,-1.425],[0.436,-0.383],[0,0],[9.529,9.397]],"o":[[0,0],[-0.542,-0.487],[-1.866,-1.722],[-6.406,-6.266],[-19.739,-22.371],[-19.146,-37.773],[4.485,-50.144],[50.086,-42.341],[47.221,3.006],[25.28,20.534],[15.222,-20.401],[30.424,-24.711],[48.264,-3.073],[48.974,41.395],[10.396,43.936],[-12.985,32.927],[-7.782,9.213],[-2.604,2.542],[-0.622,0.563],[0,0],[-9.529,9.397],[0,0]],"v":[[-284.573,97.974],[-284.622,97.93],[-286.845,95.904],[-294.535,88.584],[-319.71,62.021],[-383.67,-30.448],[-413.106,-163.021],[-333.514,-305.328],[-184.822,-361.596],[-67.552,-312.007],[-6.532,-247.313],[54.486,-312.006],[171.757,-361.596],[320.45,-305.329],[407.082,-175.984],[397.234,-58.463],[349.699,19.47],[329.671,41.044],[323.328,47.014],[321.737,48.434],[10.655,355.271],[-23.716,355.271]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.792156862745,0.027450980392,0.623529411765,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":6,"op":62,"st":1,"bm":0},{"ddd":0,"ind":2,"ty":4,"nm":"Like Color","parent":5,"tt":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[0,0,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.5,"y":1},"o":{"x":0.19,"y":0},"t":0,"s":[{"i":[[0,0],[0,0],[0.934,0.862],[3.208,3.137],[9.951,11.279],[19.234,37.944],[-4.372,48.876],[-50.933,43.057],[-48.268,-3.073],[-30.424,-24.711],[-15.223,-20.401],[-25.28,20.533],[-47.222,3.006],[-50.09,-42.34],[-10.435,-44.105],[13.083,-33.182],[15.403,-18.231],[5.198,-5.073],[1.581,-1.425],[0.436,-0.383],[0,0],[9.529,9.397]],"o":[[0,0],[-0.542,-0.487],[-1.866,-1.722],[-6.406,-6.266],[-19.739,-22.371],[-19.146,-37.773],[4.485,-50.144],[50.086,-42.341],[47.221,3.006],[25.28,20.534],[15.222,-20.401],[30.424,-24.711],[48.264,-3.073],[48.974,41.395],[10.396,43.936],[-12.985,32.927],[-7.782,9.213],[-2.604,2.542],[-0.622,0.563],[0,0],[-9.529,9.397],[0,0]],"v":[[-284.573,97.974],[-284.622,97.93],[-286.845,95.904],[-294.535,88.584],[-319.71,62.021],[-383.67,-30.448],[-413.106,-163.021],[-333.514,-305.328],[-184.822,-361.596],[-67.552,-312.007],[-6.532,-247.313],[54.486,-312.006],[171.757,-361.596],[320.45,-305.329],[407.082,-175.984],[397.234,-58.463],[349.699,19.47],[329.671,41.044],[323.328,47.014],[321.737,48.434],[10.655,355.271],[-23.716,355.271]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":5,"s":[{"i":[[15.537,10.805],[0,0],[0,0],[0,0],[0,0],[24.175,34.947],[-4.877,38.506],[-56.821,33.922],[-53.848,-2.421],[-33.941,-19.469],[-16.983,-16.073],[-28.203,16.177],[-52.681,2.369],[-55.881,-33.356],[-11.642,-34.748],[13.993,-26.374],[0,0],[0,0],[1.764,-1.123],[0,0],[204.541,-93.271],[15.378,6.376]],"o":[[-15.537,-10.805],[-0.604,-0.383],[0,0],[0,0],[0,0],[-20.785,-30.046],[5.004,-39.505],[55.876,-33.358],[52.68,2.368],[28.203,16.177],[16.981,-16.073],[33.941,-19.469],[53.844,-2.421],[54.635,32.613],[11.597,34.614],[-31.498,59.367],[0,0],[0,0],[-0.694,0.444],[0,0],[-11.329,3.018],[-202.799,-88.104]],"v":[[-303.824,211.323],[-330.653,190.152],[-350.982,173.525],[-368.486,159.303],[-397.687,131.331],[-435.015,80.555],[-465.623,-46.906],[-371.81,-163.249],[-205.928,-207.579],[-75.101,-168.51],[-7.027,-117.542],[61.045,-168.51],[191.873,-207.579],[357.756,-163.25],[454.403,-61.346],[449.552,41.575],[348.551,159.808],[313.936,186.669],[294.309,201.001],[271.896,217.15],[12.146,357.199],[-26.197,357.199]],"c":true}]},{"i":{"x":0.833,"y":1},"o":{"x":0.167,"y":0},"t":12,"s":[{"i":[[0,0],[0,0],[0.819,0.913],[2.814,3.325],[8.731,11.953],[16.875,40.213],[-3.836,51.798],[-44.687,45.632],[-42.348,-3.257],[-26.693,-26.189],[-13.356,-21.621],[-22.18,21.761],[-41.431,3.186],[-43.948,-44.871],[-9.156,-46.742],[11.481,-35.165],[13.512,-19.323],[4.56,-5.377],[1.387,-1.51],[0.382,-0.406],[0,0],[8.36,9.959]],"o":[[0,0],[-0.475,-0.516],[-1.637,-1.825],[-5.62,-6.641],[-17.318,-23.709],[-16.798,-40.031],[3.935,-53.142],[43.943,-44.873],[41.43,3.186],[22.18,21.761],[13.355,-21.621],[26.693,-26.189],[42.345,-3.257],[42.968,43.87],[9.121,46.563],[-11.393,34.896],[-6.828,9.764],[-2.285,2.694],[-0.545,0.597],[0,0],[-8.361,9.959],[0,0]],"v":[[-249.674,103.832],[-249.717,103.786],[-251.667,101.638],[-258.414,93.88],[-280.502,65.729],[-336.618,-32.269],[-362.443,-172.769],[-292.613,-323.585],[-162.156,-383.217],[-59.268,-330.662],[-5.731,-262.1],[47.804,-330.662],[150.693,-383.217],[281.151,-323.586],[357.159,-186.507],[348.519,-61.958],[306.813,20.634],[289.241,43.499],[283.676,49.825],[282.28,51.33],[9.348,376.514],[-20.807,376.514]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.167,"y":0},"t":17,"s":[{"i":[[0,0],[0,0],[0.934,0.862],[3.208,3.137],[9.951,11.279],[19.234,37.944],[-4.372,48.876],[-50.933,43.057],[-48.268,-3.073],[-30.424,-24.711],[-15.223,-20.401],[-25.28,20.533],[-47.222,3.006],[-50.09,-42.34],[-10.435,-44.105],[13.083,-33.182],[15.403,-18.231],[5.198,-5.073],[1.581,-1.425],[0.436,-0.383],[0,0],[9.529,9.397]],"o":[[0,0],[-0.542,-0.487],[-1.866,-1.722],[-6.406,-6.266],[-19.739,-22.371],[-19.146,-37.773],[4.485,-50.144],[50.086,-42.341],[47.221,3.006],[25.28,20.534],[15.222,-20.401],[30.424,-24.711],[48.264,-3.073],[48.974,41.395],[10.396,43.936],[-12.985,32.927],[-7.782,9.213],[-2.604,2.542],[-0.622,0.563],[0,0],[-9.529,9.397],[0,0]],"v":[[-284.573,97.974],[-284.622,97.93],[-286.845,95.904],[-294.535,88.584],[-319.71,62.021],[-383.67,-30.448],[-413.106,-163.021],[-333.514,-305.328],[-184.822,-361.596],[-67.552,-312.007],[-6.532,-247.313],[54.486,-312.006],[171.757,-361.596],[320.45,-305.329],[407.082,-175.984],[397.234,-58.463],[349.699,19.47],[329.671,41.044],[323.328,47.014],[321.737,48.434],[10.655,355.271],[-23.716,355.271]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":20,"s":[{"i":[[14.574,11.501],[0,0],[0,0],[0,0],[0,0],[22.678,37.202],[-4.575,40.99],[-53.302,36.11],[-50.513,-2.577],[-31.839,-20.725],[-15.931,-17.11],[-26.456,17.221],[-49.418,2.521],[-52.421,-35.508],[-10.921,-36.989],[13.127,-28.076],[0,0],[0,0],[1.654,-1.195],[0,0],[191.873,-99.288],[14.425,6.788]],"o":[[-14.574,-11.502],[-0.567,-0.408],[0,0],[0,0],[0,0],[-19.497,-31.985],[4.694,-42.054],[52.415,-35.51],[49.418,2.521],[26.456,17.221],[15.93,-17.11],[31.839,-20.725],[50.509,-2.577],[51.252,34.716],[10.879,36.847],[-29.547,63.196],[0,0],[0,0],[-0.651,0.473],[0,0],[-10.627,3.212],[-190.239,-93.788]],"v":[[-285.147,201.501],[-310.314,178.965],[-329.384,161.265],[-345.804,146.126],[-373.197,116.349],[-408.213,62.298],[-436.925,-73.386],[-348.922,-197.233],[-193.314,-244.423],[-70.589,-202.834],[-6.731,-148.578],[57.125,-202.834],[179.85,-244.423],[335.46,-197.234],[426.121,-88.757],[421.571,20.804],[326.825,146.664],[294.354,175.257],[275.942,190.513],[254.917,207.705],[21.255,356.575],[-42.714,356.575]],"c":true}]},{"t":24,"s":[{"i":[[0,0],[0,0],[0.934,0.862],[3.208,3.137],[9.951,11.279],[19.234,37.944],[-4.372,48.876],[-50.933,43.057],[-48.268,-3.073],[-30.424,-24.711],[-15.223,-20.401],[-25.28,20.533],[-47.222,3.006],[-50.09,-42.34],[-10.435,-44.105],[13.083,-33.182],[15.403,-18.231],[5.198,-5.073],[1.581,-1.425],[0.436,-0.383],[0,0],[9.529,9.397]],"o":[[0,0],[-0.542,-0.487],[-1.866,-1.722],[-6.406,-6.266],[-19.739,-22.371],[-19.146,-37.773],[4.485,-50.144],[50.086,-42.341],[47.221,3.006],[25.28,20.534],[15.222,-20.401],[30.424,-24.711],[48.264,-3.073],[48.974,41.395],[10.396,43.936],[-12.985,32.927],[-7.782,9.213],[-2.604,2.542],[-0.622,0.563],[0,0],[-9.529,9.397],[0,0]],"v":[[-284.573,97.974],[-284.622,97.93],[-286.845,95.904],[-294.535,88.584],[-319.71,62.021],[-383.67,-30.448],[-413.106,-163.021],[-333.514,-305.328],[-184.822,-361.596],[-67.552,-312.007],[-6.532,-247.313],[54.486,-312.006],[171.757,-361.596],[320.45,-305.329],[407.082,-175.984],[397.234,-58.463],[349.699,19.47],[329.671,41.044],[323.328,47.014],[321.737,48.434],[10.655,355.271],[-23.716,355.271]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.792156862745,0.027450980392,0.623529411765,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":6,"op":62,"st":1,"bm":0},{"ddd":0,"ind":3,"ty":4,"nm":"Like Color 4","parent":5,"td":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":5,"s":[-2,434.828,0],"to":[0.333,-72.471,0],"ti":[-0.333,72.471,0]},{"t":12,"s":[0,0,0]}],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.4,0.4,0.4],"y":[1,1,1]},"o":{"x":[0.6,0.6,0.6],"y":[0,0,0]},"t":5,"s":[36,36,100]},{"t":12,"s":[100,100,100]}],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.5,"y":1},"o":{"x":0.19,"y":0},"t":0,"s":[{"i":[[0,0],[0,0],[0.934,0.862],[3.208,3.137],[9.951,11.279],[19.234,37.944],[-4.372,48.876],[-50.933,43.057],[-48.268,-3.073],[-30.424,-24.711],[-15.223,-20.401],[-25.28,20.533],[-47.222,3.006],[-50.09,-42.34],[-10.435,-44.105],[13.083,-33.182],[15.403,-18.231],[5.198,-5.073],[1.581,-1.425],[0.436,-0.383],[0,0],[9.529,9.397]],"o":[[0,0],[-0.542,-0.487],[-1.866,-1.722],[-6.406,-6.266],[-19.739,-22.371],[-19.146,-37.773],[4.485,-50.144],[50.086,-42.341],[47.221,3.006],[25.28,20.534],[15.222,-20.401],[30.424,-24.711],[48.264,-3.073],[48.974,41.395],[10.396,43.936],[-12.985,32.927],[-7.782,9.213],[-2.604,2.542],[-0.622,0.563],[0,0],[-9.529,9.397],[0,0]],"v":[[-284.573,97.974],[-284.622,97.93],[-286.845,95.904],[-294.535,88.584],[-319.71,62.021],[-383.67,-30.448],[-413.106,-163.021],[-333.514,-305.328],[-184.822,-361.596],[-67.552,-312.007],[-6.532,-247.313],[54.486,-312.006],[171.757,-361.596],[320.45,-305.329],[407.082,-175.984],[397.234,-58.463],[349.699,19.47],[329.671,41.044],[323.328,47.014],[321.737,48.434],[10.655,355.271],[-23.716,355.271]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":5,"s":[{"i":[[15.537,10.805],[0,0],[0,0],[0,0],[0,0],[24.175,34.947],[-4.877,38.506],[-56.821,33.922],[-53.848,-2.421],[-33.941,-19.469],[-16.983,-16.073],[-28.203,16.177],[-52.681,2.369],[-55.881,-33.356],[-11.642,-34.748],[13.993,-26.374],[0,0],[0,0],[1.764,-1.123],[0,0],[204.541,-93.271],[15.378,6.376]],"o":[[-15.537,-10.805],[-0.604,-0.383],[0,0],[0,0],[0,0],[-20.785,-30.046],[5.004,-39.505],[55.876,-33.358],[52.68,2.368],[28.203,16.177],[16.981,-16.073],[33.941,-19.469],[53.844,-2.421],[54.635,32.613],[11.597,34.614],[-31.498,59.367],[0,0],[0,0],[-0.694,0.444],[0,0],[-11.329,3.018],[-202.799,-88.104]],"v":[[-303.824,211.323],[-330.653,190.152],[-350.982,173.525],[-368.486,159.303],[-397.687,131.331],[-435.015,80.555],[-465.623,-46.906],[-371.81,-163.249],[-205.928,-207.579],[-75.101,-168.51],[-7.027,-117.542],[61.045,-168.51],[191.873,-207.579],[357.756,-163.25],[454.403,-61.346],[449.552,41.575],[348.551,159.808],[313.936,186.669],[294.309,201.001],[271.896,217.15],[12.146,357.199],[-26.197,357.199]],"c":true}]},{"i":{"x":0.833,"y":1},"o":{"x":0.167,"y":0},"t":12,"s":[{"i":[[0,0],[0,0],[0.819,0.913],[2.814,3.325],[8.731,11.953],[16.875,40.213],[-3.836,51.798],[-44.687,45.632],[-42.348,-3.257],[-26.693,-26.189],[-13.356,-21.621],[-22.18,21.761],[-41.431,3.186],[-43.948,-44.871],[-9.156,-46.742],[11.481,-35.165],[13.512,-19.323],[4.56,-5.377],[1.387,-1.51],[0.382,-0.406],[0,0],[8.36,9.959]],"o":[[0,0],[-0.475,-0.516],[-1.637,-1.825],[-5.62,-6.641],[-17.318,-23.709],[-16.798,-40.031],[3.935,-53.142],[43.943,-44.873],[41.43,3.186],[22.18,21.761],[13.355,-21.621],[26.693,-26.189],[42.345,-3.257],[42.968,43.87],[9.121,46.563],[-11.393,34.896],[-6.828,9.764],[-2.285,2.694],[-0.545,0.597],[0,0],[-8.361,9.959],[0,0]],"v":[[-249.674,103.832],[-249.717,103.786],[-251.667,101.638],[-258.414,93.88],[-280.502,65.729],[-336.618,-32.269],[-362.443,-172.769],[-292.613,-323.585],[-162.156,-383.217],[-59.268,-330.662],[-5.731,-262.1],[47.804,-330.662],[150.693,-383.217],[281.151,-323.586],[357.159,-186.507],[348.519,-61.958],[306.813,20.634],[289.241,43.499],[283.676,49.825],[282.28,51.33],[9.348,376.514],[-20.807,376.514]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.167,"y":0},"t":17,"s":[{"i":[[0,0],[0,0],[0.934,0.862],[3.208,3.137],[9.951,11.279],[19.234,37.944],[-4.372,48.876],[-50.933,43.057],[-48.268,-3.073],[-30.424,-24.711],[-15.223,-20.401],[-25.28,20.533],[-47.222,3.006],[-50.09,-42.34],[-10.435,-44.105],[13.083,-33.182],[15.403,-18.231],[5.198,-5.073],[1.581,-1.425],[0.436,-0.383],[0,0],[9.529,9.397]],"o":[[0,0],[-0.542,-0.487],[-1.866,-1.722],[-6.406,-6.266],[-19.739,-22.371],[-19.146,-37.773],[4.485,-50.144],[50.086,-42.341],[47.221,3.006],[25.28,20.534],[15.222,-20.401],[30.424,-24.711],[48.264,-3.073],[48.974,41.395],[10.396,43.936],[-12.985,32.927],[-7.782,9.213],[-2.604,2.542],[-0.622,0.563],[0,0],[-9.529,9.397],[0,0]],"v":[[-284.573,97.974],[-284.622,97.93],[-286.845,95.904],[-294.535,88.584],[-319.71,62.021],[-383.67,-30.448],[-413.106,-163.021],[-333.514,-305.328],[-184.822,-361.596],[-67.552,-312.007],[-6.532,-247.313],[54.486,-312.006],[171.757,-361.596],[320.45,-305.329],[407.082,-175.984],[397.234,-58.463],[349.699,19.47],[329.671,41.044],[323.328,47.014],[321.737,48.434],[10.655,355.271],[-23.716,355.271]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":20,"s":[{"i":[[14.574,11.501],[0,0],[0,0],[0,0],[0,0],[22.678,37.202],[-4.575,40.99],[-53.302,36.11],[-50.513,-2.577],[-31.839,-20.725],[-15.931,-17.11],[-26.456,17.221],[-49.418,2.521],[-52.421,-35.508],[-10.921,-36.989],[13.127,-28.076],[0,0],[0,0],[1.654,-1.195],[0,0],[191.873,-99.288],[14.425,6.788]],"o":[[-14.574,-11.502],[-0.567,-0.408],[0,0],[0,0],[0,0],[-19.497,-31.985],[4.694,-42.054],[52.415,-35.51],[49.418,2.521],[26.456,17.221],[15.93,-17.11],[31.839,-20.725],[50.509,-2.577],[51.252,34.716],[10.879,36.847],[-29.547,63.196],[0,0],[0,0],[-0.651,0.473],[0,0],[-10.627,3.212],[-190.239,-93.788]],"v":[[-285.147,201.501],[-310.314,178.965],[-329.384,161.265],[-345.804,146.126],[-373.197,116.349],[-408.213,62.298],[-436.925,-73.386],[-348.922,-197.233],[-193.314,-244.423],[-70.589,-202.834],[-6.731,-148.578],[57.125,-202.834],[179.85,-244.423],[335.46,-197.234],[426.121,-88.757],[421.571,20.804],[326.825,146.664],[294.354,175.257],[275.942,190.513],[254.917,207.705],[21.255,356.575],[-42.714,356.575]],"c":true}]},{"t":24,"s":[{"i":[[0,0],[0,0],[0.934,0.862],[3.208,3.137],[9.951,11.279],[19.234,37.944],[-4.372,48.876],[-50.933,43.057],[-48.268,-3.073],[-30.424,-24.711],[-15.223,-20.401],[-25.28,20.533],[-47.222,3.006],[-50.09,-42.34],[-10.435,-44.105],[13.083,-33.182],[15.403,-18.231],[5.198,-5.073],[1.581,-1.425],[0.436,-0.383],[0,0],[9.529,9.397]],"o":[[0,0],[-0.542,-0.487],[-1.866,-1.722],[-6.406,-6.266],[-19.739,-22.371],[-19.146,-37.773],[4.485,-50.144],[50.086,-42.341],[47.221,3.006],[25.28,20.534],[15.222,-20.401],[30.424,-24.711],[48.264,-3.073],[48.974,41.395],[10.396,43.936],[-12.985,32.927],[-7.782,9.213],[-2.604,2.542],[-0.622,0.563],[0,0],[-9.529,9.397],[0,0]],"v":[[-284.573,97.974],[-284.622,97.93],[-286.845,95.904],[-294.535,88.584],[-319.71,62.021],[-383.67,-30.448],[-413.106,-163.021],[-333.514,-305.328],[-184.822,-361.596],[-67.552,-312.007],[-6.532,-247.313],[54.486,-312.006],[171.757,-361.596],[320.45,-305.329],[407.082,-175.984],[397.234,-58.463],[349.699,19.47],[329.671,41.044],[323.328,47.014],[321.737,48.434],[10.655,355.271],[-23.716,355.271]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.792156862745,0.027450980392,0.623529411765,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":5,"op":61,"st":0,"bm":0},{"ddd":0,"ind":4,"ty":4,"nm":"Like Color 3","parent":5,"tt":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[0,0,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.5,"y":1},"o":{"x":0.19,"y":0},"t":0,"s":[{"i":[[0,0],[0,0],[0.934,0.862],[3.208,3.137],[9.951,11.279],[19.234,37.944],[-4.372,48.876],[-50.933,43.057],[-48.268,-3.073],[-30.424,-24.711],[-15.223,-20.401],[-25.28,20.533],[-47.222,3.006],[-50.09,-42.34],[-10.435,-44.105],[13.083,-33.182],[15.403,-18.231],[5.198,-5.073],[1.581,-1.425],[0.436,-0.383],[0,0],[9.529,9.397]],"o":[[0,0],[-0.542,-0.487],[-1.866,-1.722],[-6.406,-6.266],[-19.739,-22.371],[-19.146,-37.773],[4.485,-50.144],[50.086,-42.341],[47.221,3.006],[25.28,20.534],[15.222,-20.401],[30.424,-24.711],[48.264,-3.073],[48.974,41.395],[10.396,43.936],[-12.985,32.927],[-7.782,9.213],[-2.604,2.542],[-0.622,0.563],[0,0],[-9.529,9.397],[0,0]],"v":[[-284.573,97.974],[-284.622,97.93],[-286.845,95.904],[-294.535,88.584],[-319.71,62.021],[-383.67,-30.448],[-413.106,-163.021],[-333.514,-305.328],[-184.822,-361.596],[-67.552,-312.007],[-6.532,-247.313],[54.486,-312.006],[171.757,-361.596],[320.45,-305.329],[407.082,-175.984],[397.234,-58.463],[349.699,19.47],[329.671,41.044],[323.328,47.014],[321.737,48.434],[10.655,355.271],[-23.716,355.271]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":5,"s":[{"i":[[15.537,10.805],[0,0],[0,0],[0,0],[0,0],[24.175,34.947],[-4.877,38.506],[-56.821,33.922],[-53.848,-2.421],[-33.941,-19.469],[-16.983,-16.073],[-28.203,16.177],[-52.681,2.369],[-55.881,-33.356],[-11.642,-34.748],[13.993,-26.374],[0,0],[0,0],[1.764,-1.123],[0,0],[204.541,-93.271],[15.378,6.376]],"o":[[-15.537,-10.805],[-0.604,-0.383],[0,0],[0,0],[0,0],[-20.785,-30.046],[5.004,-39.505],[55.876,-33.358],[52.68,2.368],[28.203,16.177],[16.981,-16.073],[33.941,-19.469],[53.844,-2.421],[54.635,32.613],[11.597,34.614],[-31.498,59.367],[0,0],[0,0],[-0.694,0.444],[0,0],[-11.329,3.018],[-202.799,-88.104]],"v":[[-303.824,211.323],[-330.653,190.152],[-350.982,173.525],[-368.486,159.303],[-397.687,131.331],[-435.015,80.555],[-465.623,-46.906],[-371.81,-163.249],[-205.928,-207.579],[-75.101,-168.51],[-7.027,-117.542],[61.045,-168.51],[191.873,-207.579],[357.756,-163.25],[454.403,-61.346],[449.552,41.575],[348.551,159.808],[313.936,186.669],[294.309,201.001],[271.896,217.15],[12.146,357.199],[-26.197,357.199]],"c":true}]},{"i":{"x":0.833,"y":1},"o":{"x":0.167,"y":0},"t":12,"s":[{"i":[[0,0],[0,0],[0.819,0.913],[2.814,3.325],[8.731,11.953],[16.875,40.213],[-3.836,51.798],[-44.687,45.632],[-42.348,-3.257],[-26.693,-26.189],[-13.356,-21.621],[-22.18,21.761],[-41.431,3.186],[-43.948,-44.871],[-9.156,-46.742],[11.481,-35.165],[13.512,-19.323],[4.56,-5.377],[1.387,-1.51],[0.382,-0.406],[0,0],[8.36,9.959]],"o":[[0,0],[-0.475,-0.516],[-1.637,-1.825],[-5.62,-6.641],[-17.318,-23.709],[-16.798,-40.031],[3.935,-53.142],[43.943,-44.873],[41.43,3.186],[22.18,21.761],[13.355,-21.621],[26.693,-26.189],[42.345,-3.257],[42.968,43.87],[9.121,46.563],[-11.393,34.896],[-6.828,9.764],[-2.285,2.694],[-0.545,0.597],[0,0],[-8.361,9.959],[0,0]],"v":[[-249.674,103.832],[-249.717,103.786],[-251.667,101.638],[-258.414,93.88],[-280.502,65.729],[-336.618,-32.269],[-362.443,-172.769],[-292.613,-323.585],[-162.156,-383.217],[-59.268,-330.662],[-5.731,-262.1],[47.804,-330.662],[150.693,-383.217],[281.151,-323.586],[357.159,-186.507],[348.519,-61.958],[306.813,20.634],[289.241,43.499],[283.676,49.825],[282.28,51.33],[9.348,376.514],[-20.807,376.514]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.167,"y":0},"t":17,"s":[{"i":[[0,0],[0,0],[0.934,0.862],[3.208,3.137],[9.951,11.279],[19.234,37.944],[-4.372,48.876],[-50.933,43.057],[-48.268,-3.073],[-30.424,-24.711],[-15.223,-20.401],[-25.28,20.533],[-47.222,3.006],[-50.09,-42.34],[-10.435,-44.105],[13.083,-33.182],[15.403,-18.231],[5.198,-5.073],[1.581,-1.425],[0.436,-0.383],[0,0],[9.529,9.397]],"o":[[0,0],[-0.542,-0.487],[-1.866,-1.722],[-6.406,-6.266],[-19.739,-22.371],[-19.146,-37.773],[4.485,-50.144],[50.086,-42.341],[47.221,3.006],[25.28,20.534],[15.222,-20.401],[30.424,-24.711],[48.264,-3.073],[48.974,41.395],[10.396,43.936],[-12.985,32.927],[-7.782,9.213],[-2.604,2.542],[-0.622,0.563],[0,0],[-9.529,9.397],[0,0]],"v":[[-284.573,97.974],[-284.622,97.93],[-286.845,95.904],[-294.535,88.584],[-319.71,62.021],[-383.67,-30.448],[-413.106,-163.021],[-333.514,-305.328],[-184.822,-361.596],[-67.552,-312.007],[-6.532,-247.313],[54.486,-312.006],[171.757,-361.596],[320.45,-305.329],[407.082,-175.984],[397.234,-58.463],[349.699,19.47],[329.671,41.044],[323.328,47.014],[321.737,48.434],[10.655,355.271],[-23.716,355.271]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":20,"s":[{"i":[[14.574,11.501],[0,0],[0,0],[0,0],[0,0],[22.678,37.202],[-4.575,40.99],[-53.302,36.11],[-50.513,-2.577],[-31.839,-20.725],[-15.931,-17.11],[-26.456,17.221],[-49.418,2.521],[-52.421,-35.508],[-10.921,-36.989],[13.127,-28.076],[0,0],[0,0],[1.654,-1.195],[0,0],[191.873,-99.288],[14.425,6.788]],"o":[[-14.574,-11.502],[-0.567,-0.408],[0,0],[0,0],[0,0],[-19.497,-31.985],[4.694,-42.054],[52.415,-35.51],[49.418,2.521],[26.456,17.221],[15.93,-17.11],[31.839,-20.725],[50.509,-2.577],[51.252,34.716],[10.879,36.847],[-29.547,63.196],[0,0],[0,0],[-0.651,0.473],[0,0],[-10.627,3.212],[-190.239,-93.788]],"v":[[-285.147,201.501],[-310.314,178.965],[-329.384,161.265],[-345.804,146.126],[-373.197,116.349],[-408.213,62.298],[-436.925,-73.386],[-348.922,-197.233],[-193.314,-244.423],[-70.589,-202.834],[-6.731,-148.578],[57.125,-202.834],[179.85,-244.423],[335.46,-197.234],[426.121,-88.757],[421.571,20.804],[326.825,146.664],[294.354,175.257],[275.942,190.513],[254.917,207.705],[21.255,356.575],[-42.714,356.575]],"c":true}]},{"t":24,"s":[{"i":[[0,0],[0,0],[0.934,0.862],[3.208,3.137],[9.951,11.279],[19.234,37.944],[-4.372,48.876],[-50.933,43.057],[-48.268,-3.073],[-30.424,-24.711],[-15.223,-20.401],[-25.28,20.533],[-47.222,3.006],[-50.09,-42.34],[-10.435,-44.105],[13.083,-33.182],[15.403,-18.231],[5.198,-5.073],[1.581,-1.425],[0.436,-0.383],[0,0],[9.529,9.397]],"o":[[0,0],[-0.542,-0.487],[-1.866,-1.722],[-6.406,-6.266],[-19.739,-22.371],[-19.146,-37.773],[4.485,-50.144],[50.086,-42.341],[47.221,3.006],[25.28,20.534],[15.222,-20.401],[30.424,-24.711],[48.264,-3.073],[48.974,41.395],[10.396,43.936],[-12.985,32.927],[-7.782,9.213],[-2.604,2.542],[-0.622,0.563],[0,0],[-9.529,9.397],[0,0]],"v":[[-284.573,97.974],[-284.622,97.93],[-286.845,95.904],[-294.535,88.584],[-319.71,62.021],[-383.67,-30.448],[-413.106,-163.021],[-333.514,-305.328],[-184.822,-361.596],[-67.552,-312.007],[-6.532,-247.313],[54.486,-312.006],[171.757,-361.596],[320.45,-305.329],[407.082,-175.984],[397.234,-58.463],[349.699,19.47],[329.671,41.044],[323.328,47.014],[321.737,48.434],[10.655,355.271],[-23.716,355.271]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.874509803922,0.531564749923,0.798880722943,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":5,"op":61,"st":0,"bm":0},{"ddd":0,"ind":5,"ty":4,"nm":"Like","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":5,"s":[0]},{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":12,"s":[13]},{"t":17,"s":[0]}],"ix":10},"p":{"a":1,"k":[{"i":{"x":0,"y":1},"o":{"x":1,"y":0},"t":5,"s":[800,800,0],"to":[1.667,-22.833,0],"ti":[0,0,0]},{"i":{"x":0.999,"y":1},"o":{"x":0.8,"y":0},"t":12,"s":[810,663,0],"to":[0,0,0],"ti":[1.667,-22.833,0]},{"t":17,"s":[800,800,0]}],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.4,0.4,0.4],"y":[1,1,1]},"o":{"x":[0.6,0.6,0.6],"y":[0,0,0]},"t":5,"s":[100,100,100]},{"i":{"x":[0.4,0.4,0.4],"y":[1,1,1]},"o":{"x":[0.6,0.6,0.6],"y":[0,0,0]},"t":12,"s":[121,121,100]},{"t":17,"s":[100,100,100]}],"ix":6}},"ao":0,"hasMask":true,"masksProperties":[{"inv":false,"mode":"s","pt":{"a":1,"k":[{"i":{"x":0.5,"y":1},"o":{"x":0.19,"y":0},"t":0,"s":[{"i":[[0,0],[0,0],[9.287,18.318],[0,0],[0,0],[0.398,0.734],[1.686,2.893],[6.63,9.544],[23.055,18.726],[26.012,1.656],[38.106,-32.213],[2.601,-29.08],[-15.501,-30.58],[-17.271,-19.574],[-5.308,-5.191],[-1.369,-1.263],[-0.306,-0.275],[0,0],[0,0],[0,0],[0,0],[-9.529,9.397],[0,0],[0,0],[0,0],[0,0],[0,0],[-1.87,1.828],[-5.996,7.099],[-9.069,22.999],[6.143,25.967],[39.224,33.159],[24.971,-1.59],[23.126,-18.782],[13.435,-19.34],[3.378,-5.799],[0.795,-1.47],[0.166,-0.318]],"o":[[0,0],[-9.286,18.318],[0,0],[0,0],[-0.166,-0.318],[-0.796,-1.469],[-3.379,-5.799],[-13.437,-19.34],[-23.124,-18.782],[-24.966,-1.59],[-37.259,31.498],[-2.715,30.347],[15.414,30.409],[8.553,9.694],[2.649,2.592],[0.684,0.631],[0,0],[0,0],[0,0],[0,0],[9.529,9.397],[0,0],[0,0],[0,0],[0,0],[0,0],[0.92,-0.833],[3.759,-3.666],[12.163,-14.394],[8.967,-22.744],[-6.103,-25.797],[-38.113,-32.214],[-26.009,1.656],[-23.053,18.726],[-6.629,9.544],[-1.686,2.893],[-0.397,0.734],[0,0]],"v":[[27.765,-160.404],[16.508,-138.198],[-28.568,-138.197],[-39.807,-160.366],[-39.983,-160.704],[-40.829,-162.292],[-44.561,-168.912],[-59.654,-192.529],[-115.107,-253.946],[-189.191,-286.686],[-283.819,-248.212],[-337.301,-155.968],[-315.553,-63.293],[-262.184,13.571],[-240.802,36.139],[-234.701,41.952],[-233.206,43.315],[-232.905,43.584],[-232.862,43.622],[-232.076,44.311],[-23.216,250.321],[11.154,250.321],[270.655,-5.637],[271.658,-6.485],[271.707,-6.527],[271.8,-6.609],[272.759,-7.461],[277.008,-11.469],[292.122,-27.763],[327.279,-84.988],[333.964,-158.54],[271.761,-248.211],[177.122,-286.686],[103.039,-253.947],[47.59,-192.53],[32.5,-168.913],[28.769,-162.293],[27.922,-160.706]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":5,"s":[{"i":[[0,0],[0,0],[10.158,13.194],[0,0],[0,0],[0.435,0.529],[1.844,2.084],[7.253,6.874],[25.217,13.488],[28.451,1.193],[41.679,-23.202],[2.846,-20.945],[-9.356,-17.439],[-11.826,-12.205],[0,0],[0,0],[-0.335,-0.198],[0,0],[0,0],[0,0],[-65.391,-27.115],[-16.238,6.813],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[-10.258,10.459],[-7.899,16.306],[6.72,18.703],[42.905,23.881],[27.313,-1.145],[25.294,-13.53],[14.694,-13.932],[3.696,-4.176],[0.87,-1.059],[0.182,-0.229]],"o":[[0,0],[-10.157,13.194],[0,0],[0,0],[-0.182,-0.229],[-0.87,-1.058],[-3.696,-4.177],[-14.697,-13.93],[-25.293,-13.528],[-27.307,-1.145],[-40.753,22.687],[-2.97,21.858],[9.356,17.439],[13.327,13.754],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[16.096,7.029],[73.661,-26.654],[0,0],[0,0],[0,0],[0,0],[1.006,-0.6],[0,0],[10.258,-10.459],[7.899,-16.306],[-6.676,-18.581],[-41.687,-23.203],[-28.448,1.193],[-25.215,13.488],[-7.25,6.874],[-1.844,2.083],[-0.435,0.529],[0,0]],"v":[[30.365,-47.434],[18.052,-36.054],[-31.251,-36.054],[-43.545,-47.407],[-43.737,-47.651],[-44.663,-48.794],[-48.744,-53.562],[-65.253,-70.573],[-125.907,-109.273],[-206.938,-132.855],[-310.441,-105.143],[-368.939,-38.702],[-351.714,36.354],[-317.403,82.027],[-281.438,115.355],[-263.826,128.308],[-251.253,137.134],[-237.798,146.556],[-214.782,160.887],[-176.186,184.454],[-24.304,265.471],[13.29,265.471],[199.233,163.243],[232.051,142.33],[251.792,127.996],[268.301,115.479],[288.491,97.332],[302.982,85.217],[330.451,58.254],[360.702,17.037],[365.28,-40.555],[297.243,-105.143],[193.729,-132.855],[112.699,-109.274],[52.049,-70.574],[35.544,-53.563],[31.463,-48.795],[30.536,-47.652]],"c":true}]},{"i":{"x":0.833,"y":1},"o":{"x":0.167,"y":0},"t":12,"s":[{"i":[[0,0],[0,0],[8.613,20.725],[0,0],[0,0],[0.369,0.831],[1.563,3.273],[6.149,10.798],[21.381,21.187],[24.123,1.874],[35.339,-36.447],[2.413,-32.901],[-14.376,-34.599],[-16.016,-22.147],[-4.921,-5.874],[-1.27,-1.429],[-0.284,-0.311],[0,0],[0,0],[0,0],[0,0],[-8.837,10.632],[0,0],[0,0],[0,0],[0,0],[0,0],[-1.736,2.066],[-5.562,8.031],[-8.409,26.022],[5.698,29.379],[36.38,37.513],[23.159,-1.799],[21.446,-21.252],[12.458,-21.883],[3.134,-6.56],[0.738,-1.663],[0.154,-0.359]],"o":[[0,0],[-8.612,20.726],[0,0],[0,0],[-0.154,-0.359],[-0.738,-1.662],[-3.133,-6.561],[-12.461,-21.882],[-21.446,-21.251],[-23.154,-1.799],[-34.554,35.637],[-2.518,34.335],[14.295,34.405],[7.932,10.968],[2.457,2.932],[0.635,0.714],[0,0],[0,0],[0,0],[0,0],[8.837,10.632],[0,0],[0,0],[0,0],[0,0],[0,0],[0.853,-0.942],[3.486,-4.148],[11.28,-16.286],[8.316,-25.733],[-5.66,-29.187],[-35.346,-36.447],[-24.121,1.873],[-21.379,21.186],[-6.147,10.798],[-1.563,3.273],[-0.369,0.831],[0,0]],"v":[[25.753,-179.606],[15.313,-154.482],[-26.491,-154.481],[-36.914,-179.563],[-37.077,-179.946],[-37.862,-181.742],[-41.323,-189.232],[-55.32,-215.952],[-106.748,-285.44],[-175.454,-322.483],[-263.213,-278.953],[-312.813,-174.587],[-292.644,-69.734],[-243.149,17.23],[-223.319,42.765],[-217.66,49.341],[-216.274,50.883],[-215.995,51.188],[-215.955,51.231],[-215.226,52.01],[-21.527,285.091],[10.348,285.091],[251.011,-4.501],[251.941,-5.461],[251.987,-5.509],[252.073,-5.601],[252.963,-6.565],[256.902,-11.1],[270.919,-29.535],[303.524,-94.28],[309.725,-177.497],[252.037,-278.952],[164.267,-322.483],[95.563,-285.441],[44.139,-215.953],[30.144,-189.233],[26.684,-181.743],[25.898,-179.947]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.167,"y":0},"t":17,"s":[{"i":[[0,0],[0,0],[9.287,18.318],[0,0],[0,0],[0.398,0.734],[1.686,2.893],[6.63,9.544],[23.055,18.726],[26.012,1.656],[38.106,-32.213],[2.601,-29.08],[-15.501,-30.58],[-17.271,-19.574],[-5.308,-5.191],[-1.369,-1.263],[-0.306,-0.275],[0,0],[0,0],[0,0],[0,0],[-9.529,9.397],[0,0],[0,0],[0,0],[0,0],[0,0],[-1.87,1.828],[-5.996,7.099],[-9.069,22.999],[6.143,25.967],[39.224,33.159],[24.971,-1.59],[23.126,-18.782],[13.435,-19.34],[3.378,-5.799],[0.795,-1.47],[0.166,-0.318]],"o":[[0,0],[-9.286,18.318],[0,0],[0,0],[-0.166,-0.318],[-0.796,-1.469],[-3.379,-5.799],[-13.437,-19.34],[-23.124,-18.782],[-24.966,-1.59],[-37.259,31.498],[-2.715,30.347],[15.414,30.409],[8.553,9.694],[2.649,2.592],[0.684,0.631],[0,0],[0,0],[0,0],[0,0],[9.529,9.397],[0,0],[0,0],[0,0],[0,0],[0,0],[0.92,-0.833],[3.759,-3.666],[12.163,-14.394],[8.967,-22.744],[-6.103,-25.797],[-38.113,-32.214],[-26.009,1.656],[-23.053,18.726],[-6.629,9.544],[-1.686,2.893],[-0.397,0.734],[0,0]],"v":[[27.765,-160.404],[16.508,-138.198],[-28.568,-138.197],[-39.807,-160.366],[-39.983,-160.704],[-40.829,-162.292],[-44.561,-168.912],[-59.654,-192.529],[-115.107,-253.946],[-189.191,-286.686],[-283.819,-248.212],[-337.301,-155.968],[-315.553,-63.293],[-262.184,13.571],[-240.802,36.139],[-234.701,41.952],[-233.206,43.315],[-232.905,43.584],[-232.862,43.622],[-232.076,44.311],[-23.216,250.321],[11.154,250.321],[270.655,-5.637],[271.658,-6.485],[271.707,-6.527],[271.8,-6.609],[272.759,-7.461],[277.008,-11.469],[292.122,-27.763],[327.279,-84.988],[333.964,-158.54],[271.761,-248.211],[177.122,-286.686],[103.039,-253.947],[47.59,-192.53],[32.5,-168.913],[28.769,-162.293],[27.922,-160.706]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":20,"s":[{"i":[[0,0],[0,0],[9.721,14.297],[0,0],[0,0],[0.416,0.573],[1.764,2.258],[6.941,7.449],[24.132,14.616],[27.227,1.293],[39.886,-25.143],[2.723,-22.697],[-8.953,-18.897],[-11.318,-13.226],[0,0],[0,0],[-0.32,-0.215],[0,0],[0,0],[0,0],[-62.578,-29.383],[-15.539,7.383],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[-9.816,11.334],[-7.559,17.67],[6.431,20.267],[41.06,25.878],[26.138,-1.241],[24.206,-14.662],[14.062,-15.097],[3.537,-4.525],[0.833,-1.147],[0.174,-0.248]],"o":[[0,0],[-9.72,14.298],[0,0],[0,0],[-0.174,-0.248],[-0.833,-1.147],[-3.537,-4.526],[-14.065,-15.095],[-24.205,-14.66],[-26.133,-1.241],[-39,24.584],[-2.842,23.686],[8.953,18.897],[12.754,14.904],[0,0],[0,0],[0,0],[0,0],[0,0],[0,0],[15.404,7.617],[70.493,-28.883],[0,0],[0,0],[0,0],[0,0],[0.963,-0.65],[0,0],[9.816,-11.334],[7.559,-17.67],[-6.389,-20.135],[-39.894,-25.144],[-27.225,1.292],[-24.13,14.616],[-6.938,7.449],[-1.764,2.258],[-0.416,0.573],[0,0]],"v":[[29.061,-74.194],[17.278,-61.862],[-29.905,-61.861],[-41.669,-74.164],[-41.853,-74.428],[-42.74,-75.667],[-46.646,-80.834],[-62.444,-99.268],[-120.489,-141.205],[-198.035,-166.759],[-297.085,-136.73],[-353.067,-64.732],[-336.583,16.603],[-303.748,66.096],[-269.329,102.211],[-252.475,116.248],[-240.443,125.812],[-227.567,136.022],[-205.541,151.552],[-168.605,177.09],[-23.256,264.883],[12.72,264.883],[190.665,154.105],[222.071,131.442],[240.963,115.909],[256.762,102.345],[276.083,82.681],[289.951,69.552],[316.239,40.334],[345.188,-4.33],[349.569,-66.739],[284.459,-136.729],[185.397,-166.759],[107.853,-141.206],[49.812,-99.269],[34.017,-80.835],[30.111,-75.668],[29.225,-74.429]],"c":true}]},{"t":24,"s":[{"i":[[0,0],[0,0],[9.287,18.318],[0,0],[0,0],[0.398,0.734],[1.686,2.893],[6.63,9.544],[23.055,18.726],[26.012,1.656],[38.106,-32.213],[2.601,-29.08],[-15.501,-30.58],[-17.271,-19.574],[-5.308,-5.191],[-1.369,-1.263],[-0.306,-0.275],[0,0],[0,0],[0,0],[0,0],[-9.529,9.397],[0,0],[0,0],[0,0],[0,0],[0,0],[-1.87,1.828],[-5.996,7.099],[-9.069,22.999],[6.143,25.967],[39.224,33.159],[24.971,-1.59],[23.126,-18.782],[13.435,-19.34],[3.378,-5.799],[0.795,-1.47],[0.166,-0.318]],"o":[[0,0],[-9.286,18.318],[0,0],[0,0],[-0.166,-0.318],[-0.796,-1.469],[-3.379,-5.799],[-13.437,-19.34],[-23.124,-18.782],[-24.966,-1.59],[-37.259,31.498],[-2.715,30.347],[15.414,30.409],[8.553,9.694],[2.649,2.592],[0.684,0.631],[0,0],[0,0],[0,0],[0,0],[9.529,9.397],[0,0],[0,0],[0,0],[0,0],[0,0],[0.92,-0.833],[3.759,-3.666],[12.163,-14.394],[8.967,-22.744],[-6.103,-25.797],[-38.113,-32.214],[-26.009,1.656],[-23.053,18.726],[-6.629,9.544],[-1.686,2.893],[-0.397,0.734],[0,0]],"v":[[27.765,-160.404],[16.508,-138.198],[-28.568,-138.197],[-39.807,-160.366],[-39.983,-160.704],[-40.829,-162.292],[-44.561,-168.912],[-59.654,-192.529],[-115.107,-253.946],[-189.191,-286.686],[-283.819,-248.212],[-337.301,-155.968],[-315.553,-63.293],[-262.184,13.571],[-240.802,36.139],[-234.701,41.952],[-233.206,43.315],[-232.905,43.584],[-232.862,43.622],[-232.076,44.311],[-23.216,250.321],[11.154,250.321],[270.655,-5.637],[271.658,-6.485],[271.707,-6.527],[271.8,-6.609],[272.759,-7.461],[277.008,-11.469],[292.122,-27.763],[327.279,-84.988],[333.964,-158.54],[271.761,-248.211],[177.122,-286.686],[103.039,-253.947],[47.59,-192.53],[32.5,-168.913],[28.769,-162.293],[27.922,-160.706]],"c":true}]}],"ix":1},"o":{"a":0,"k":100,"ix":3},"x":{"a":0,"k":0,"ix":4},"nm":"Mask 1"}],"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.5,"y":1},"o":{"x":0.19,"y":0},"t":0,"s":[{"i":[[0,0],[0,0],[0.934,0.862],[3.208,3.137],[9.951,11.279],[19.234,37.944],[-4.372,48.876],[-50.933,43.057],[-48.268,-3.073],[-30.424,-24.711],[-15.223,-20.401],[-25.28,20.533],[-47.222,3.006],[-50.09,-42.34],[-10.435,-44.105],[13.083,-33.182],[15.403,-18.231],[5.198,-5.073],[1.581,-1.425],[0.436,-0.383],[0,0],[9.529,9.397]],"o":[[0,0],[-0.542,-0.487],[-1.866,-1.722],[-6.406,-6.266],[-19.739,-22.371],[-19.146,-37.773],[4.485,-50.144],[50.086,-42.341],[47.221,3.006],[25.28,20.534],[15.222,-20.401],[30.424,-24.711],[48.264,-3.073],[48.974,41.395],[10.396,43.936],[-12.985,32.927],[-7.782,9.213],[-2.604,2.542],[-0.622,0.563],[0,0],[-9.529,9.397],[0,0]],"v":[[-284.573,97.974],[-284.622,97.93],[-286.845,95.904],[-294.535,88.584],[-319.71,62.021],[-383.67,-30.448],[-413.106,-163.021],[-333.514,-305.328],[-184.822,-361.596],[-67.552,-312.007],[-6.532,-247.313],[54.486,-312.006],[171.757,-361.596],[320.45,-305.329],[407.082,-175.984],[397.234,-58.463],[349.699,19.47],[329.671,41.044],[323.328,47.014],[321.737,48.434],[10.655,355.271],[-23.716,355.271]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":5,"s":[{"i":[[15.537,10.804],[0,0],[0,0],[0,0],[0,0],[24.175,34.947],[-4.877,38.506],[-56.821,33.922],[-53.848,-2.421],[-33.941,-19.469],[-16.983,-16.073],[-28.203,16.177],[-52.681,2.369],[-55.881,-33.356],[-11.642,-34.748],[13.993,-26.374],[0,0],[0,0],[1.764,-1.123],[0,0],[204.541,-93.271],[15.378,6.376]],"o":[[-15.537,-10.805],[-0.604,-0.383],[0,0],[0,0],[0,0],[-20.785,-30.046],[5.004,-39.505],[55.876,-33.358],[52.68,2.368],[28.203,16.177],[16.981,-16.073],[33.941,-19.469],[53.844,-2.421],[54.635,32.613],[11.597,34.614],[-31.498,59.367],[0,0],[0,0],[-0.694,0.444],[0,0],[-11.329,3.018],[-202.799,-88.104]],"v":[[-303.824,211.323],[-330.653,190.152],[-350.982,173.525],[-368.486,159.303],[-397.687,131.331],[-435.015,80.555],[-465.623,-46.906],[-371.81,-163.249],[-205.928,-207.579],[-75.101,-168.51],[-7.027,-117.542],[61.045,-168.51],[191.873,-207.579],[357.756,-163.25],[454.403,-61.346],[449.552,41.575],[348.551,159.808],[313.936,186.669],[294.309,201.001],[271.896,217.15],[30.646,357.295],[-51.197,357.295]],"c":true}]},{"i":{"x":0.833,"y":1},"o":{"x":0.167,"y":0},"t":12,"s":[{"i":[[0,0],[0,0],[0.819,0.913],[2.814,3.325],[8.731,11.953],[16.875,40.213],[-3.836,51.798],[-44.687,45.632],[-42.348,-3.257],[-26.693,-26.189],[-13.356,-21.621],[-22.18,21.761],[-41.431,3.186],[-43.948,-44.871],[-9.156,-46.742],[11.481,-35.165],[13.512,-19.323],[4.56,-5.377],[1.387,-1.51],[0.382,-0.406],[0,0],[8.36,9.959]],"o":[[0,0],[-0.475,-0.516],[-1.637,-1.825],[-5.62,-6.641],[-17.318,-23.709],[-16.798,-40.031],[3.935,-53.142],[43.943,-44.873],[41.43,3.186],[22.18,21.761],[13.355,-21.621],[26.693,-26.189],[42.345,-3.257],[42.968,43.87],[9.121,46.563],[-11.393,34.896],[-6.828,9.764],[-2.285,2.694],[-0.545,0.597],[0,0],[-8.361,9.959],[0,0]],"v":[[-249.674,103.832],[-249.717,103.786],[-251.667,101.638],[-258.414,93.88],[-280.502,65.729],[-336.618,-32.269],[-362.443,-172.769],[-292.613,-323.585],[-162.156,-383.217],[-59.268,-330.662],[-5.731,-262.1],[47.804,-330.662],[150.693,-383.217],[281.151,-323.586],[357.159,-186.507],[348.519,-61.958],[306.813,20.634],[289.241,43.499],[283.676,49.825],[282.28,51.33],[9.348,376.514],[-20.807,376.514]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.167,"y":0},"t":17,"s":[{"i":[[0,0],[0,0],[0.934,0.862],[3.208,3.137],[9.951,11.279],[19.234,37.944],[-4.372,48.876],[-50.933,43.057],[-48.268,-3.073],[-30.424,-24.711],[-15.223,-20.401],[-25.28,20.533],[-47.222,3.006],[-50.09,-42.34],[-10.435,-44.105],[13.083,-33.182],[15.403,-18.231],[5.198,-5.073],[1.581,-1.425],[0.436,-0.383],[0,0],[9.529,9.397]],"o":[[0,0],[-0.542,-0.487],[-1.866,-1.722],[-6.406,-6.266],[-19.739,-22.371],[-19.146,-37.773],[4.485,-50.144],[50.086,-42.341],[47.221,3.006],[25.28,20.534],[15.222,-20.401],[30.424,-24.711],[48.264,-3.073],[48.974,41.395],[10.396,43.936],[-12.985,32.927],[-7.782,9.213],[-2.604,2.542],[-0.622,0.563],[0,0],[-9.529,9.397],[0,0]],"v":[[-284.573,97.974],[-284.622,97.93],[-286.845,95.904],[-294.535,88.584],[-319.71,62.021],[-383.67,-30.448],[-413.106,-163.021],[-333.514,-305.328],[-184.822,-361.596],[-67.552,-312.007],[-6.532,-247.313],[54.486,-312.006],[171.757,-361.596],[320.45,-305.329],[407.082,-175.984],[397.234,-58.463],[349.699,19.47],[329.671,41.044],[323.328,47.014],[321.737,48.434],[10.655,355.271],[-23.716,355.271]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":20,"s":[{"i":[[14.574,11.501],[0,0],[0,0],[0,0],[0,0],[22.678,37.202],[-4.575,40.99],[-53.302,36.11],[-50.513,-2.577],[-31.839,-20.725],[-15.931,-17.11],[-26.456,17.221],[-49.418,2.521],[-52.421,-35.508],[-10.921,-36.989],[13.127,-28.076],[0,0],[0,0],[1.654,-1.195],[0,0],[191.873,-99.288],[14.425,6.788]],"o":[[-14.574,-11.502],[-0.567,-0.408],[0,0],[0,0],[0,0],[-19.497,-31.985],[4.694,-42.054],[52.415,-35.51],[49.418,2.521],[26.456,17.221],[15.93,-17.11],[31.839,-20.725],[50.509,-2.577],[51.252,34.716],[10.879,36.847],[-29.547,63.196],[0,0],[0,0],[-0.651,0.473],[0,0],[-10.627,3.212],[-190.239,-93.788]],"v":[[-285.147,201.501],[-310.314,178.965],[-329.384,161.265],[-345.804,146.126],[-373.197,116.349],[-408.213,62.298],[-436.925,-73.386],[-348.922,-197.233],[-193.314,-244.423],[-70.589,-202.834],[-6.731,-148.578],[57.125,-202.834],[179.85,-244.423],[335.46,-197.234],[426.121,-88.757],[421.571,20.804],[326.825,146.664],[294.354,175.257],[275.942,190.513],[254.917,207.705],[21.255,356.575],[-42.714,356.575]],"c":true}]},{"t":24,"s":[{"i":[[0,0],[0,0],[0.934,0.862],[3.208,3.137],[9.951,11.279],[19.234,37.944],[-4.372,48.876],[-50.933,43.057],[-48.268,-3.073],[-30.424,-24.711],[-15.223,-20.401],[-25.28,20.533],[-47.222,3.006],[-50.09,-42.34],[-10.435,-44.105],[13.083,-33.182],[15.403,-18.231],[5.198,-5.073],[1.581,-1.425],[0.436,-0.383],[0,0],[9.529,9.397]],"o":[[0,0],[-0.542,-0.487],[-1.866,-1.722],[-6.406,-6.266],[-19.739,-22.371],[-19.146,-37.773],[4.485,-50.144],[50.086,-42.341],[47.221,3.006],[25.28,20.534],[15.222,-20.401],[30.424,-24.711],[48.264,-3.073],[48.974,41.395],[10.396,43.936],[-12.985,32.927],[-7.782,9.213],[-2.604,2.542],[-0.622,0.563],[0,0],[-9.529,9.397],[0,0]],"v":[[-284.573,97.974],[-284.622,97.93],[-286.845,95.904],[-294.535,88.584],[-319.71,62.021],[-383.67,-30.448],[-413.106,-163.021],[-333.514,-305.328],[-184.822,-361.596],[-67.552,-312.007],[-6.532,-247.313],[54.486,-312.006],[171.757,-361.596],[320.45,-305.329],[407.082,-175.984],[397.234,-58.463],[349.699,19.47],[329.671,41.044],[323.328,47.014],[321.737,48.434],[10.655,355.271],[-23.716,355.271]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.4,0.4,0.4,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":61,"st":0,"bm":0},{"ddd":0,"ind":6,"ty":4,"nm":"Like Afterglow","parent":5,"sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[45.556]},{"t":20,"s":[0]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[0,0,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.854]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0]},"t":9,"s":[100,100,100]},{"i":{"x":[0.999,0.999,0.999],"y":[1,1,1]},"o":{"x":[0.6,0.6,0.6],"y":[0,0,0]},"t":13,"s":[115.222,115.222,100]},{"t":20,"s":[168,168,100]}],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.5,"y":1},"o":{"x":0.19,"y":0},"t":0,"s":[{"i":[[0,0],[0,0],[0.934,0.862],[3.208,3.137],[9.951,11.279],[19.234,37.944],[-4.372,48.876],[-50.933,43.057],[-48.268,-3.073],[-30.424,-24.711],[-15.223,-20.401],[-25.28,20.533],[-47.222,3.006],[-50.09,-42.34],[-10.435,-44.105],[13.083,-33.182],[15.403,-18.231],[5.198,-5.073],[1.581,-1.425],[0.436,-0.383],[0,0],[9.529,9.397]],"o":[[0,0],[-0.542,-0.487],[-1.866,-1.722],[-6.406,-6.266],[-19.739,-22.371],[-19.146,-37.773],[4.485,-50.144],[50.086,-42.341],[47.221,3.006],[25.28,20.534],[15.222,-20.401],[30.424,-24.711],[48.264,-3.073],[48.974,41.395],[10.396,43.936],[-12.985,32.927],[-7.782,9.213],[-2.604,2.542],[-0.622,0.563],[0,0],[-9.529,9.397],[0,0]],"v":[[-284.573,97.974],[-284.622,97.93],[-286.845,95.904],[-294.535,88.584],[-319.71,62.021],[-383.67,-30.448],[-413.106,-163.021],[-333.514,-305.328],[-184.822,-361.596],[-67.552,-312.007],[-6.532,-247.313],[54.486,-312.006],[171.757,-361.596],[320.45,-305.329],[407.082,-175.984],[397.234,-58.463],[349.699,19.47],[329.671,41.044],[323.328,47.014],[321.737,48.434],[10.655,355.271],[-23.716,355.271]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":5,"s":[{"i":[[15.537,10.805],[0,0],[0,0],[0,0],[0,0],[24.175,34.947],[-4.877,38.506],[-56.821,33.922],[-53.848,-2.421],[-33.941,-19.469],[-16.983,-16.073],[-28.203,16.177],[-52.681,2.369],[-55.881,-33.356],[-11.642,-34.748],[13.993,-26.374],[0,0],[0,0],[1.764,-1.123],[0,0],[204.541,-93.271],[15.378,6.376]],"o":[[-15.537,-10.805],[-0.604,-0.383],[0,0],[0,0],[0,0],[-20.785,-30.046],[5.004,-39.505],[55.876,-33.358],[52.68,2.368],[28.203,16.177],[16.981,-16.073],[33.941,-19.469],[53.844,-2.421],[54.635,32.613],[11.597,34.614],[-31.498,59.367],[0,0],[0,0],[-0.694,0.444],[0,0],[-11.329,3.018],[-202.799,-88.104]],"v":[[-303.824,211.323],[-330.653,190.152],[-350.982,173.525],[-368.486,159.303],[-397.687,131.331],[-435.015,80.555],[-465.623,-46.906],[-371.81,-163.249],[-205.928,-207.579],[-75.101,-168.51],[-7.027,-117.542],[61.045,-168.51],[191.873,-207.579],[357.756,-163.25],[454.403,-61.346],[449.552,41.575],[348.551,159.808],[313.936,186.669],[294.309,201.001],[271.896,217.15],[12.146,357.199],[-26.197,357.199]],"c":true}]},{"i":{"x":0.833,"y":1},"o":{"x":0.167,"y":0},"t":12,"s":[{"i":[[0,0],[0,0],[0.819,0.913],[2.814,3.325],[8.731,11.953],[16.875,40.213],[-3.836,51.798],[-44.687,45.632],[-42.348,-3.257],[-26.693,-26.189],[-13.356,-21.621],[-22.18,21.761],[-41.431,3.186],[-43.948,-44.871],[-9.156,-46.742],[11.481,-35.165],[13.512,-19.323],[4.56,-5.377],[1.387,-1.51],[0.382,-0.406],[0,0],[8.36,9.959]],"o":[[0,0],[-0.475,-0.516],[-1.637,-1.825],[-5.62,-6.641],[-17.318,-23.709],[-16.798,-40.031],[3.935,-53.142],[43.943,-44.873],[41.43,3.186],[22.18,21.761],[13.355,-21.621],[26.693,-26.189],[42.345,-3.257],[42.968,43.87],[9.121,46.563],[-11.393,34.896],[-6.828,9.764],[-2.285,2.694],[-0.545,0.597],[0,0],[-8.361,9.959],[0,0]],"v":[[-249.674,103.832],[-249.717,103.786],[-251.667,101.638],[-258.414,93.88],[-280.502,65.729],[-336.618,-32.269],[-362.443,-172.769],[-292.613,-323.585],[-162.156,-383.217],[-59.268,-330.662],[-5.731,-262.1],[47.804,-330.662],[150.693,-383.217],[281.151,-323.586],[357.159,-186.507],[348.519,-61.958],[306.813,20.634],[289.241,43.499],[283.676,49.825],[282.28,51.33],[9.348,376.514],[-20.807,376.514]],"c":true}]},{"i":{"x":0.5,"y":1},"o":{"x":0.167,"y":0},"t":17,"s":[{"i":[[0,0],[0,0],[0.934,0.862],[3.208,3.137],[9.951,11.279],[19.234,37.944],[-4.372,48.876],[-50.933,43.057],[-48.268,-3.073],[-30.424,-24.711],[-15.223,-20.401],[-25.28,20.533],[-47.222,3.006],[-50.09,-42.34],[-10.435,-44.105],[13.083,-33.182],[15.403,-18.231],[5.198,-5.073],[1.581,-1.425],[0.436,-0.383],[0,0],[9.529,9.397]],"o":[[0,0],[-0.542,-0.487],[-1.866,-1.722],[-6.406,-6.266],[-19.739,-22.371],[-19.146,-37.773],[4.485,-50.144],[50.086,-42.341],[47.221,3.006],[25.28,20.534],[15.222,-20.401],[30.424,-24.711],[48.264,-3.073],[48.974,41.395],[10.396,43.936],[-12.985,32.927],[-7.782,9.213],[-2.604,2.542],[-0.622,0.563],[0,0],[-9.529,9.397],[0,0]],"v":[[-284.573,97.974],[-284.622,97.93],[-286.845,95.904],[-294.535,88.584],[-319.71,62.021],[-383.67,-30.448],[-413.106,-163.021],[-333.514,-305.328],[-184.822,-361.596],[-67.552,-312.007],[-6.532,-247.313],[54.486,-312.006],[171.757,-361.596],[320.45,-305.329],[407.082,-175.984],[397.234,-58.463],[349.699,19.47],[329.671,41.044],[323.328,47.014],[321.737,48.434],[10.655,355.271],[-23.716,355.271]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":20,"s":[{"i":[[14.574,11.501],[0,0],[0,0],[0,0],[0,0],[22.678,37.202],[-4.575,40.99],[-53.302,36.11],[-50.513,-2.577],[-31.839,-20.725],[-15.931,-17.11],[-26.456,17.221],[-49.418,2.521],[-52.421,-35.508],[-10.921,-36.989],[13.127,-28.076],[0,0],[0,0],[1.654,-1.195],[0,0],[191.873,-99.288],[14.425,6.788]],"o":[[-14.574,-11.502],[-0.567,-0.408],[0,0],[0,0],[0,0],[-19.497,-31.985],[4.694,-42.054],[52.415,-35.51],[49.418,2.521],[26.456,17.221],[15.93,-17.11],[31.839,-20.725],[50.509,-2.577],[51.252,34.716],[10.879,36.847],[-29.547,63.196],[0,0],[0,0],[-0.651,0.473],[0,0],[-10.627,3.212],[-190.239,-93.788]],"v":[[-285.147,201.501],[-310.314,178.965],[-329.384,161.265],[-345.804,146.126],[-373.197,116.349],[-408.213,62.298],[-436.925,-73.386],[-348.922,-197.233],[-193.314,-244.423],[-70.589,-202.834],[-6.731,-148.578],[57.125,-202.834],[179.85,-244.423],[335.46,-197.234],[426.121,-88.757],[421.571,20.804],[326.825,146.664],[294.354,175.257],[275.942,190.513],[254.917,207.705],[21.255,356.575],[-42.714,356.575]],"c":true}]},{"t":24,"s":[{"i":[[0,0],[0,0],[0.934,0.862],[3.208,3.137],[9.951,11.279],[19.234,37.944],[-4.372,48.876],[-50.933,43.057],[-48.268,-3.073],[-30.424,-24.711],[-15.223,-20.401],[-25.28,20.533],[-47.222,3.006],[-50.09,-42.34],[-10.435,-44.105],[13.083,-33.182],[15.403,-18.231],[5.198,-5.073],[1.581,-1.425],[0.436,-0.383],[0,0],[9.529,9.397]],"o":[[0,0],[-0.542,-0.487],[-1.866,-1.722],[-6.406,-6.266],[-19.739,-22.371],[-19.146,-37.773],[4.485,-50.144],[50.086,-42.341],[47.221,3.006],[25.28,20.534],[15.222,-20.401],[30.424,-24.711],[48.264,-3.073],[48.974,41.395],[10.396,43.936],[-12.985,32.927],[-7.782,9.213],[-2.604,2.542],[-0.622,0.563],[0,0],[-9.529,9.397],[0,0]],"v":[[-284.573,97.974],[-284.622,97.93],[-286.845,95.904],[-294.535,88.584],[-319.71,62.021],[-383.67,-30.448],[-413.106,-163.021],[-333.514,-305.328],[-184.822,-361.596],[-67.552,-312.007],[-6.532,-247.313],[54.486,-312.006],[171.757,-361.596],[320.45,-305.329],[407.082,-175.984],[397.234,-58.463],[349.699,19.47],[329.671,41.044],[323.328,47.014],[321.737,48.434],[10.655,355.271],[-23.716,355.271]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,0.607843137255,0.913524373372,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":12,"op":62,"st":1,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/src/assets/lottie/zap_md.json b/src/assets/lottie/zap_md.json new file mode 100644 index 0000000..917ce01 --- /dev/null +++ b/src/assets/lottie/zap_md.json @@ -0,0 +1 @@ +{"v":"4.8.0","meta":{"g":"LottieFiles AE 3.4.5","a":"","k":"","d":"","tc":""},"fr":30,"ip":0,"op":31,"w":1125,"h":300,"nm":"Zap Medium","ddd":0,"assets":[{"id":"image_0","w":1125,"h":2436,"u":"","p":"","e":1},{"id":"image_1","w":1125,"h":2436,"u":"","p":"","e":1},{"id":"image_2","w":1125,"h":2436,"u":"","p":"","e":1},{"id":"image_3","w":1125,"h":2436,"u":"","p":"","e":1},{"id":"image_4","w":1125,"h":2436,"u":"","p":"","e":1},{"id":"image_5","w":1125,"h":2436,"u":"","p":"","e":1},{"id":"image_6","w":1125,"h":2436,"u":"","p":"","e":1},{"id":"image_7","w":1125,"h":2436,"u":"","p":"","e":1},{"id":"image_8","w":1125,"h":2436,"u":"","p":"","e":1},{"id":"image_9","w":1125,"h":2436,"u":"","p":"","e":1},{"id":"image_10","w":1125,"h":2436,"u":"","p":"","e":1},{"id":"image_11","w":1125,"h":2436,"u":"","p":"","e":1},{"id":"image_12","w":1125,"h":2436,"u":"","p":"","e":1},{"id":"image_13","w":1125,"h":2436,"u":"","p":"","e":1},{"id":"image_14","w":1125,"h":2436,"u":"","p":"","e":1},{"id":"image_15","w":1125,"h":2436,"u":"","p":"","e":1},{"id":"image_16","w":1125,"h":2436,"u":"","p":"","e":1},{"id":"image_17","w":1125,"h":2436,"u":"","p":"","e":1},{"id":"image_18","w":1125,"h":2436,"u":"","p":"","e":1}],"layers":[{"ddd":0,"ind":1,"ty":2,"nm":"0.ai","cl":"ai","refId":"image_0","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[562.5,-372,0],"ix":2},"a":{"a":0,"k":[562.5,1218,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":0,"k":255,"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":31,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"ip":0,"op":0.5,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":2,"nm":"1.ai","cl":"ai","refId":"image_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[562.5,-372,0],"ix":2},"a":{"a":0,"k":[562.5,1218,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":0,"k":255,"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":31,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"ip":0.5,"op":1,"st":0.5,"bm":0},{"ddd":0,"ind":3,"ty":2,"nm":"2.ai","cl":"ai","refId":"image_2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[562.5,-372,0],"ix":2},"a":{"a":0,"k":[562.5,1218,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":0,"k":255,"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":31,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"ip":1,"op":1.5,"st":1,"bm":0},{"ddd":0,"ind":4,"ty":2,"nm":"3.ai","cl":"ai","refId":"image_3","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[562.5,-372,0],"ix":2},"a":{"a":0,"k":[562.5,1218,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":0,"k":255,"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":31,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"ip":1.5,"op":2,"st":1.5,"bm":0},{"ddd":0,"ind":5,"ty":2,"nm":"4.ai","cl":"ai","refId":"image_4","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[562.5,-372,0],"ix":2},"a":{"a":0,"k":[562.5,1218,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":0,"k":255,"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":31,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"ip":2,"op":2.5,"st":2,"bm":0},{"ddd":0,"ind":6,"ty":2,"nm":"5.ai","cl":"ai","refId":"image_5","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[562.5,-372,0],"ix":2},"a":{"a":0,"k":[562.5,1218,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":0,"k":255,"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":31,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"ip":2.5,"op":3,"st":2.5,"bm":0},{"ddd":0,"ind":7,"ty":2,"nm":"6.ai","cl":"ai","refId":"image_6","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[562.5,-372,0],"ix":2},"a":{"a":0,"k":[562.5,1218,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":0,"k":255,"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":31,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"ip":3,"op":3.5,"st":3,"bm":0},{"ddd":0,"ind":8,"ty":2,"nm":"7.ai","cl":"ai","refId":"image_7","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[562.5,-372,0],"ix":2},"a":{"a":0,"k":[562.5,1218,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":0,"k":255,"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":31,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"ip":3.5,"op":4,"st":3.5,"bm":0},{"ddd":0,"ind":9,"ty":2,"nm":"8.ai","cl":"ai","refId":"image_8","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[562.5,-372,0],"ix":2},"a":{"a":0,"k":[562.5,1218,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":0,"k":255,"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":31,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"ip":4,"op":4.5,"st":4,"bm":0},{"ddd":0,"ind":10,"ty":2,"nm":"9.ai","cl":"ai","refId":"image_9","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[562.5,-372,0],"ix":2},"a":{"a":0,"k":[562.5,1218,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":0,"k":255,"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":31,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"ip":4.5,"op":5,"st":4.5,"bm":0},{"ddd":0,"ind":11,"ty":2,"nm":"10.ai","cl":"ai","refId":"image_10","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[562.5,-372,0],"ix":2},"a":{"a":0,"k":[562.5,1218,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":0,"k":255,"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":31,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"ip":6,"op":6.5,"st":6,"bm":0},{"ddd":0,"ind":12,"ty":2,"nm":"11.ai","cl":"ai","refId":"image_11","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[562.5,-372,0],"ix":2},"a":{"a":0,"k":[562.5,1218,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":0,"k":255,"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":31,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"ip":7.5,"op":8,"st":7.5,"bm":0},{"ddd":0,"ind":13,"ty":2,"nm":"12.ai","cl":"ai","refId":"image_12","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[562.5,-372,0],"ix":2},"a":{"a":0,"k":[562.5,1218,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":0,"k":255,"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":31,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"ip":9,"op":9.5,"st":9,"bm":0},{"ddd":0,"ind":14,"ty":2,"nm":"13.ai","cl":"ai","refId":"image_13","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[562.5,-372,0],"ix":2},"a":{"a":0,"k":[562.5,1218,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":0,"k":255,"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":31,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"ip":10.5,"op":11,"st":10.5,"bm":0},{"ddd":0,"ind":15,"ty":2,"nm":"14.ai","cl":"ai","refId":"image_14","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[562.5,-372,0],"ix":2},"a":{"a":0,"k":[562.5,1218,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":0,"k":255,"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":31,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"ip":12,"op":12.5,"st":12,"bm":0},{"ddd":0,"ind":16,"ty":2,"nm":"15.ai","cl":"ai","refId":"image_15","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[562.5,-372,0],"ix":2},"a":{"a":0,"k":[562.5,1218,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":0,"k":255,"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":31,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"ip":13.5,"op":14,"st":13.5,"bm":0},{"ddd":0,"ind":17,"ty":2,"nm":"16.ai","cl":"ai","refId":"image_16","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[562.5,-372,0],"ix":2},"a":{"a":0,"k":[562.5,1218,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":0,"k":255,"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":31,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"ip":15,"op":15.5,"st":15,"bm":0},{"ddd":0,"ind":18,"ty":2,"nm":"17.ai","cl":"ai","refId":"image_17","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[562.5,-372,0],"ix":2},"a":{"a":0,"k":[562.5,1218,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":0,"k":255,"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":31,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"ip":16.5,"op":17,"st":16.5,"bm":0},{"ddd":0,"ind":19,"ty":2,"nm":"18.ai","cl":"ai","refId":"image_18","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[562.5,-372,0],"ix":2},"a":{"a":0,"k":[562.5,1218,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":0,"k":255,"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":31,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"ip":18,"op":18.5,"st":18,"bm":0},{"ddd":0,"ind":20,"ty":4,"nm":"Star Dash In","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[370.25,142.75,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[6.1,6.1,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":18,"s":[255]},{"t":27,"s":[0]}],"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":23,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,21.291],[-21.288,0],[0,0],[0,-21.286],[21.288,0],[0,0]],"o":[[0,-21.286],[0,0],[21.288,0],[0,21.291],[0,0],[-21.288,0]],"v":[[-399.582,214.731],[-361.038,176.187],[-173.825,176.187],[-135.281,214.731],[-173.825,253.274],[-361.038,253.274]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":4,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":5,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":6,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":7,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":8,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":9,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":10,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":11,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":15,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":17,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":18,"s":[1,1,1,1]},{"t":19,"s":[1,0.627451002598,0.184313729405,1]}],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 4","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,-21.288],[-21.288,0],[0,0],[0,21.288],[21.288,0]],"o":[[-21.288,0],[0,21.288],[0,0],[21.288,0],[0,-21.288],[0,0]],"v":[[-405.088,-44.063],[-443.632,-5.519],[-405.088,33.025],[-305.975,33.025],[-267.432,-5.519],[-305.975,-44.063]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":4,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":5,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":6,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":7,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":8,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":9,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":10,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":11,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":15,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":17,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":18,"s":[1,1,1,1]},{"t":19,"s":[1,0.627451002598,0.184313729405,1]}],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 3","np":3,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,21.288],[-21.288,0],[0,0],[0,-21.288],[21.288,0],[0,0]],"o":[[0,-21.288],[0,0],[21.288,0],[0,21.288],[0,0],[-21.288,0]],"v":[[-355.532,-225.769],[-316.988,-264.313],[-151.8,-264.313],[-113.256,-225.769],[-151.8,-187.225],[-316.988,-187.225]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":4,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":5,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":6,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":7,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":8,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":9,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":10,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":11,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":15,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":17,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":18,"s":[1,1,1,1]},{"t":19,"s":[1,0.627451002598,0.184313729405,1]}],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 2","np":3,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false}],"ip":4,"op":55,"st":0,"bm":0},{"ddd":0,"ind":21,"ty":4,"nm":"Zap In","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[370.25,142.75,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[6.1,6.1,100],"ix":6}},"ao":0,"ef":[{"ty":25,"nm":"Drop Shadow","np":8,"mn":"ADBE Drop Shadow","ix":1,"en":1,"ef":[{"ty":2,"nm":"Shadow Color","mn":"ADBE Drop Shadow-0001","ix":1,"v":{"a":0,"k":[1,0.74027967453,0.431372523308,1],"ix":1}},{"ty":0,"nm":"Opacity","mn":"ADBE Drop Shadow-0002","ix":2,"v":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":18,"s":[255]},{"t":27,"s":[0]}],"ix":2}},{"ty":0,"nm":"Direction","mn":"ADBE Drop Shadow-0003","ix":3,"v":{"a":0,"k":135,"ix":3}},{"ty":0,"nm":"Distance","mn":"ADBE Drop Shadow-0004","ix":4,"v":{"a":0,"k":1,"ix":4}},{"ty":0,"nm":"Softness","mn":"ADBE Drop Shadow-0005","ix":5,"v":{"a":0,"k":27,"ix":5}},{"ty":7,"nm":"Shadow Only","mn":"ADBE Drop Shadow-0006","ix":6,"v":{"a":0,"k":0,"ix":6}}]}],"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[10.401,-14.029],[0,0],[-8.245,44.5],[0,0],[0,0],[-10.998,13.954],[0,0],[8.972,-44.074],[0,0]],"o":[[17.889,0],[0,0],[-27.254,36.782],[0,0],[0,0],[-18.182,0],[0,0],[28.182,-35.766],[0,0],[0,0]],"v":[[415.337,-136.678],[433.231,-103.159],[43.647,422.569],[-35.441,390.471],[18.236,100.739],[-201.349,100.739],[-218.863,66.723],[167.472,-423.584],[245.656,-389.786],[194.166,-136.678]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":4,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":5,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":6,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":7,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":8,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":9,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":10,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":11,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":15,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":17,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":18,"s":[1,1,1,1]},{"t":19,"s":[1,0.627451002598,0.184313729405,1]}],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":4,"op":51,"st":-4,"bm":0},{"ddd":0,"ind":22,"ty":4,"nm":"Zap Out","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[370.25,142.75,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[6.1,6.1,100],"ix":6}},"ao":0,"hasMask":true,"masksProperties":[{"inv":false,"mode":"s","pt":{"a":0,"k":{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[101.138,-61.986],[143.265,-269.065],[-90.054,27.047],[110.668,27.047],[66.549,265.193],[309.001,-61.986]],"c":true},"ix":1},"o":{"a":0,"k":100,"ix":3},"x":{"a":0,"k":0,"ix":4},"nm":"Mask 1"}],"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[10.401,-14.029],[0,0],[-8.245,44.5],[0,0],[0,0],[-10.998,13.954],[0,0],[8.972,-44.074],[0,0]],"o":[[17.889,0],[0,0],[-27.254,36.782],[0,0],[0,0],[-18.182,0],[0,0],[28.182,-35.766],[0,0],[0,0]],"v":[[415.337,-136.678],[433.231,-103.159],[43.647,422.569],[-35.441,390.471],[18.236,100.739],[-201.349,100.739],[-218.863,66.723],[167.472,-423.584],[245.656,-389.786],[194.166,-136.678]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.4,0.4,0.4,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":4,"st":0,"bm":0},{"ddd":0,"ind":23,"ty":4,"nm":"Star Dash Out","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[370.25,142.75,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[6.1,6.1,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,21.291],[-21.288,0],[0,0],[0,-21.286],[21.288,0],[0,0]],"o":[[0,-21.286],[0,0],[21.288,0],[0,21.291],[0,0],[-21.288,0]],"v":[[-399.582,214.731],[-361.038,176.187],[-173.825,176.187],[-135.281,214.731],[-173.825,253.274],[-361.038,253.274]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.4,0.4,0.4,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 4","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[0,-21.288],[-21.288,0],[0,0],[0,21.288],[21.288,0]],"o":[[-21.288,0],[0,21.288],[0,0],[21.288,0],[0,-21.288],[0,0]],"v":[[-405.088,-44.063],[-443.632,-5.519],[-405.088,33.025],[-305.975,33.025],[-267.432,-5.519],[-305.975,-44.063]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.4,0.4,0.4,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 3","np":3,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,21.288],[-21.288,0],[0,0],[0,-21.288],[21.288,0],[0,0]],"o":[[0,-21.288],[0,0],[21.288,0],[0,21.288],[0,0],[-21.288,0]],"v":[[-355.532,-225.769],[-316.988,-264.313],[-151.8,-264.313],[-113.256,-225.769],[-151.8,-187.225],[-316.988,-187.225]],"c":true},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.4,0.4,0.4,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 2","np":3,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":4,"st":0,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/src/assets/lottie/zap_sm.json b/src/assets/lottie/zap_sm.json new file mode 100644 index 0000000..da745d4 --- /dev/null +++ b/src/assets/lottie/zap_sm.json @@ -0,0 +1 @@ +{"v":"4.8.0","meta":{"g":"LottieFiles AE 3.4.3","a":"","k":"","d":"","tc":""},"fr":30,"ip":0,"op":44,"w":1600,"h":1600,"nm":"Icon - Zap","ddd":0,"assets":[],"layers":[{"ddd":0,"ind":3,"ty":4,"nm":"Spark 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":12,"s":[0]},{"t":22,"s":[-12]}],"ix":10},"p":{"a":0,"k":[814,1190,0],"ix":2},"a":{"a":0,"k":[14,390,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[1.355,6.986],[251,27.5]],"o":[[0,0],[-37.108,-191.324],[-40.756,-4.465]],"v":[[-5,380.5],[-8.892,354.324],[-421,14.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[5]},{"t":20,"s":[1]}],"ix":2},"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":12,"s":[-355]},{"t":21,"s":[-47]}],"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1,1,1,1]},{"t":18,"s":[1,0.627451002598,0.184313729405,1]}],"ix":3},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[0]},{"t":13,"s":[100]}],"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[3]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[43]},{"t":22,"s":[0]}],"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[-30,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 6","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[6.11,3.647],[221,-82.5]],"o":[[0,0],[-173.108,-103.324],[-38.411,14.339]],"v":[[-12,395.5],[-18.892,388.324],[-519,368.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[5]},{"t":20,"s":[1]}],"ix":2},"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":12,"s":[-355]},{"t":21,"s":[-47]}],"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1,1,1,1]},{"t":18,"s":[1,0.627451002598,0.184313729405,1]}],"ix":3},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[0]},{"t":13,"s":[100]}],"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[3]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[43]},{"t":22,"s":[0]}],"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[-30,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 5","np":3,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[7.072,-0.791],[83,-132.5]],"o":[[0,0],[-149.108,16.676],[-21.765,34.746]],"v":[[-12,434.5],[-29.892,436.324],[-321,662.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[5]},{"t":20,"s":[1]}],"ix":2},"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":12,"s":[-355]},{"t":21,"s":[-47]}],"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1,1,1,1]},{"t":18,"s":[1,0.627451002598,0.184313729405,1]}],"ix":3},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[0]},{"t":13,"s":[100]}],"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[3]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[43]},{"t":22,"s":[0]}],"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[-30,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 4","np":3,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[-6.569,-2.735],[-21,-138.5]],"o":[[0,0],[68.892,28.676],[6.146,40.537]],"v":[[66,444.5],[80.108,450.324],[261,718.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[5]},{"t":20,"s":[1]}],"ix":2},"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":12,"s":[-355]},{"t":21,"s":[-47]}],"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1,1,1,1]},{"t":18,"s":[1,0.627451002598,0.184313729405,1]}],"ix":3},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[0]},{"t":13,"s":[100]}],"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[3]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[43]},{"t":22,"s":[0]}],"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 3","np":3,"cix":2,"bm":0,"ix":4,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[-0.478,-20.587],[12.253,-27.582]],"o":[[0,0],[1.054,45.342],[-16.645,37.47]],"v":[[10.212,468.795],[9.733,490.862],[-46.462,797.667]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[5]},{"t":20,"s":[1]}],"ix":2},"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":12,"s":[-355]},{"t":21,"s":[-47]}],"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1,1,1,1]},{"t":18,"s":[1,0.627451002598,0.184313729405,1]}],"ix":3},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[0]},{"t":13,"s":[100]}],"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[3]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[43]},{"t":22,"s":[0]}],"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 7","np":3,"cix":2,"bm":0,"ix":5,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[-12.773,16.152],[-209,-22.5]],"o":[[0,0],[99.892,-126.324],[40.764,4.388]],"v":[[160,273.5],[164.109,264.324],[645,164.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[5]},{"t":20,"s":[1]}],"ix":2},"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":12,"s":[-355]},{"t":21,"s":[-47]}],"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1,1,1,1]},{"t":18,"s":[1,0.627451002598,0.184313729405,1]}],"ix":3},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[0]},{"t":13,"s":[100]}],"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[3]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[43]},{"t":22,"s":[0]}],"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 2","np":3,"cix":2,"bm":0,"ix":6,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[-19.903,5.284],[-183,-132.5]],"o":[[0,0],[151.892,-40.324],[33.209,24.045]],"v":[[21,430.5],[72.108,360.324],[631,480.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[5]},{"t":20,"s":[1]}],"ix":2},"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":12,"s":[-355]},{"t":21,"s":[-47]}],"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1,1,1,1]},{"t":18,"s":[1,0.627451002598,0.184313729405,1]}],"ix":3},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[0]},{"t":13,"s":[100]}],"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[3]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[43]},{"t":22,"s":[0]}],"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":7,"mn":"ADBE Vector Group","hd":false}],"ip":15,"op":59,"st":-2,"bm":0},{"ddd":0,"ind":4,"ty":4,"nm":"Spark","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":12,"s":[0]},{"t":22,"s":[-12]}],"ix":10},"p":{"a":0,"k":[814,1190,0],"ix":2},"a":{"a":0,"k":[14,390,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[1.355,6.986],[251,27.5]],"o":[[0,0],[-37.108,-191.324],[-40.756,-4.465]],"v":[[-5,380.5],[-8.892,354.324],[-421,14.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[5]},{"t":20,"s":[1]}],"ix":2},"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":12,"s":[-336]},{"t":21,"s":[-47]}],"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[0]},{"t":13,"s":[100]}],"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[3]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[43]},{"t":22,"s":[0]}],"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[-30,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 6","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[6.11,3.647],[221,-82.5]],"o":[[0,0],[-173.108,-103.324],[-38.411,14.339]],"v":[[-12,395.5],[-18.892,388.324],[-519,368.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[5]},{"t":20,"s":[1]}],"ix":2},"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":12,"s":[-336]},{"t":21,"s":[-47]}],"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[0]},{"t":13,"s":[100]}],"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[3]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[43]},{"t":22,"s":[0]}],"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[-30,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 5","np":3,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[7.072,-0.791],[83,-132.5]],"o":[[0,0],[-149.108,16.676],[-21.765,34.746]],"v":[[-12,434.5],[-29.892,436.324],[-321,662.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[5]},{"t":20,"s":[1]}],"ix":2},"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":12,"s":[-336]},{"t":21,"s":[-47]}],"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[0]},{"t":13,"s":[100]}],"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[3]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[43]},{"t":22,"s":[0]}],"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[-30,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 4","np":3,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[-6.569,-2.735],[-21,-138.5]],"o":[[0,0],[68.892,28.676],[6.146,40.537]],"v":[[66,444.5],[80.108,450.324],[261,718.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[5]},{"t":20,"s":[1]}],"ix":2},"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":12,"s":[-336]},{"t":21,"s":[-47]}],"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[0]},{"t":13,"s":[100]}],"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[3]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[43]},{"t":22,"s":[0]}],"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 3","np":3,"cix":2,"bm":0,"ix":4,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[-0.478,-20.587],[12.253,-27.582]],"o":[[0,0],[1.054,45.342],[-16.645,37.47]],"v":[[10.212,468.795],[9.733,490.862],[-46.462,797.667]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[5]},{"t":20,"s":[1]}],"ix":2},"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":12,"s":[-336]},{"t":21,"s":[-47]}],"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[0]},{"t":13,"s":[100]}],"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[3]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[43]},{"t":22,"s":[0]}],"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 7","np":3,"cix":2,"bm":0,"ix":5,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[-12.773,16.152],[-209,-22.5]],"o":[[0,0],[99.892,-126.324],[40.764,4.388]],"v":[[160,273.5],[164.109,264.324],[645,164.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[5]},{"t":20,"s":[1]}],"ix":2},"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":12,"s":[-336]},{"t":21,"s":[-47]}],"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[0]},{"t":13,"s":[100]}],"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[3]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[43]},{"t":22,"s":[0]}],"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 2","np":3,"cix":2,"bm":0,"ix":6,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":0,"k":{"i":[[0,0],[-19.903,5.284],[-183,-132.5]],"o":[[0,0],[151.892,-40.324],[33.209,24.045]],"v":[[21,430.5],[72.108,360.324],[631,480.5]],"c":false},"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[5]},{"t":20,"s":[1]}],"ix":2},"o":{"a":1,"k":[{"i":{"x":[0.4],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":12,"s":[-336]},{"t":21,"s":[-47]}],"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false},{"ty":"st","c":{"a":0,"k":[1,1,1,1],"ix":3},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":12,"s":[0]},{"t":13,"s":[100]}],"ix":4},"w":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":13,"s":[3]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[43]},{"t":22,"s":[0]}],"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":7,"mn":"ADBE Vector Group","hd":false}],"ip":15,"op":59,"st":-2,"bm":0},{"ddd":0,"ind":5,"ty":4,"nm":"Star Dash In","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[800,800,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.167,"y":0.167},"t":25,"s":[{"i":[[0,2.35],[-2.35,0],[0,0],[0,-2.349],[2.35,0],[0,0]],"o":[[0,-2.349],[0,0],[2.35,0],[0,2.35],[0,0],[-2.35,0]],"v":[[-363.863,214.731],[-359.609,210.476],[-359.254,210.476],[-355,214.731],[-359.254,218.985],[-359.609,218.985]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.167,"y":0},"t":28,"s":[{"i":[[0,21.291],[-21.288,0],[0,0],[0,-21.286],[21.288,0],[0,0]],"o":[[0,-21.286],[0,0],[21.288,0],[0,21.291],[0,0],[-21.288,0]],"v":[[-399.582,214.731],[-361.038,176.187],[-357.825,176.187],[-319.281,214.731],[-357.825,253.274],[-361.038,253.274]],"c":true}]},{"t":33,"s":[{"i":[[0,21.291],[-21.288,0],[0,0],[0,-21.286],[21.288,0],[0,0]],"o":[[0,-21.286],[0,0],[21.288,0],[0,21.291],[0,0],[-21.288,0]],"v":[[-399.582,214.731],[-361.038,176.187],[-173.825,176.187],[-135.281,214.731],[-173.825,253.274],[-361.038,253.274]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,0.62744140625,0.184295654297,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.8,"y":0},"t":24,"s":[-60,0],"to":[10,0],"ti":[-10,0]},{"t":33,"s":[0,0]}],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":24,"s":[0]},{"t":25,"s":[100]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 4","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.167,"y":0.167},"t":24,"s":[{"i":[[0,0],[0,-2.475],[-2.475,0],[0,0],[0,2.475],[2.475,0]],"o":[[-2.475,0],[0,2.475],[0,0],[2.475,0],[0,-2.475],[0,0]],"v":[[-405.48,-10],[-409.961,-5.519],[-405.48,-1.038],[-405.583,-1.038],[-401.102,-5.519],[-405.583,-10]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.167,"y":0},"t":27,"s":[{"i":[[0,0],[0,-21.288],[-21.288,0],[0,0],[0,21.288],[21.288,0]],"o":[[-21.288,0],[0,21.288],[0,0],[21.288,0],[0,-21.288],[0,0]],"v":[[-405.088,-44.063],[-443.632,-5.519],[-405.088,33.025],[-405.975,33.025],[-367.432,-5.519],[-405.975,-44.063]],"c":true}]},{"t":32,"s":[{"i":[[0,0],[0,-21.288],[-21.288,0],[0,0],[0,21.288],[21.288,0]],"o":[[-21.288,0],[0,21.288],[0,0],[21.288,0],[0,-21.288],[0,0]],"v":[[-405.088,-44.063],[-443.632,-5.519],[-405.088,33.025],[-305.975,33.025],[-267.432,-5.519],[-305.975,-44.063]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,0.62744140625,0.184295654297,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.8,"y":0},"t":23,"s":[-60,0],"to":[10,0],"ti":[-10,0]},{"t":32,"s":[0,0]}],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":23,"s":[0]},{"t":24,"s":[100]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 3","np":3,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.167,"y":0.167},"t":23,"s":[{"i":[[0,2.889],[-2.889,0],[0,0],[0,-2.889],[2.889,0],[0,0]],"o":[[0,-2.889],[0,0],[2.889,0],[0,2.889],[0,0],[-2.889,0]],"v":[[-321.705,-225.769],[-316.475,-231],[-316.313,-231],[-311.083,-225.769],[-316.313,-220.539],[-316.475,-220.539]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.167,"y":0},"t":26,"s":[{"i":[[0,21.288],[-21.288,0],[0,0],[0,-21.288],[21.288,0],[0,0]],"o":[[0,-21.288],[0,0],[21.288,0],[0,21.288],[0,0],[-21.288,0]],"v":[[-355.532,-225.769],[-316.988,-264.313],[-315.8,-264.313],[-277.256,-225.769],[-315.8,-187.225],[-316.988,-187.225]],"c":true}]},{"t":31,"s":[{"i":[[0,21.288],[-21.288,0],[0,0],[0,-21.288],[21.288,0],[0,0]],"o":[[0,-21.288],[0,0],[21.288,0],[0,21.288],[0,0],[-21.288,0]],"v":[[-355.532,-225.769],[-316.988,-264.313],[-151.8,-264.313],[-113.256,-225.769],[-151.8,-187.225],[-316.988,-187.225]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,0.62744140625,0.184295654297,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.8,"y":0},"t":22,"s":[-60,0],"to":[10,0],"ti":[-10,0]},{"t":31,"s":[0,0]}],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":22,"s":[0]},{"t":23,"s":[100]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 2","np":3,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false}],"ip":22,"op":71,"st":16,"bm":0},{"ddd":0,"ind":6,"ty":4,"nm":"Star Dash In 2","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[800,800,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.167,"y":0.167},"t":24,"s":[{"i":[[0,2.35],[-2.35,0],[0,0],[0,-2.349],[2.35,0],[0,0]],"o":[[0,-2.349],[0,0],[2.35,0],[0,2.35],[0,0],[-2.35,0]],"v":[[-363.863,214.731],[-359.609,210.476],[-359.254,210.476],[-355,214.731],[-359.254,218.985],[-359.609,218.985]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.167,"y":0},"t":27,"s":[{"i":[[0,21.291],[-21.288,0],[0,0],[0,-21.286],[21.288,0],[0,0]],"o":[[0,-21.286],[0,0],[21.288,0],[0,21.291],[0,0],[-21.288,0]],"v":[[-399.582,214.731],[-361.038,176.187],[-357.825,176.187],[-319.281,214.731],[-357.825,253.274],[-361.038,253.274]],"c":true}]},{"t":32,"s":[{"i":[[0,21.291],[-21.288,0],[0,0],[0,-21.286],[21.288,0],[0,0]],"o":[[0,-21.286],[0,0],[21.288,0],[0,21.291],[0,0],[-21.288,0]],"v":[[-399.582,214.731],[-361.038,176.187],[-173.825,176.187],[-135.281,214.731],[-173.825,253.274],[-361.038,253.274]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.8,"y":0},"t":23,"s":[-60,0],"to":[10,0],"ti":[-10,0]},{"t":32,"s":[0,0]}],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":23,"s":[0]},{"t":24,"s":[100]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 4","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.167,"y":0.167},"t":23,"s":[{"i":[[0,0],[0,-2.475],[-2.475,0],[0,0],[0,2.475],[2.475,0]],"o":[[-2.475,0],[0,2.475],[0,0],[2.475,0],[0,-2.475],[0,0]],"v":[[-405.48,-10],[-409.961,-5.519],[-405.48,-1.038],[-405.583,-1.038],[-401.102,-5.519],[-405.583,-10]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.167,"y":0},"t":26,"s":[{"i":[[0,0],[0,-21.288],[-21.288,0],[0,0],[0,21.288],[21.288,0]],"o":[[-21.288,0],[0,21.288],[0,0],[21.288,0],[0,-21.288],[0,0]],"v":[[-405.088,-44.063],[-443.632,-5.519],[-405.088,33.025],[-405.975,33.025],[-367.432,-5.519],[-405.975,-44.063]],"c":true}]},{"t":31,"s":[{"i":[[0,0],[0,-21.288],[-21.288,0],[0,0],[0,21.288],[21.288,0]],"o":[[-21.288,0],[0,21.288],[0,0],[21.288,0],[0,-21.288],[0,0]],"v":[[-405.088,-44.063],[-443.632,-5.519],[-405.088,33.025],[-305.975,33.025],[-267.432,-5.519],[-305.975,-44.063]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.8,"y":0},"t":22,"s":[-60,0],"to":[10,0],"ti":[-10,0]},{"t":31,"s":[0,0]}],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":22,"s":[0]},{"t":23,"s":[100]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 3","np":3,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.167,"y":0.167},"t":22,"s":[{"i":[[0,2.889],[-2.889,0],[0,0],[0,-2.889],[2.889,0],[0,0]],"o":[[0,-2.889],[0,0],[2.889,0],[0,2.889],[0,0],[-2.889,0]],"v":[[-321.705,-225.769],[-316.475,-231],[-316.313,-231],[-311.083,-225.769],[-316.313,-220.539],[-316.475,-220.539]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.167,"y":0},"t":25,"s":[{"i":[[0,21.288],[-21.288,0],[0,0],[0,-21.288],[21.288,0],[0,0]],"o":[[0,-21.288],[0,0],[21.288,0],[0,21.288],[0,0],[-21.288,0]],"v":[[-355.532,-225.769],[-316.988,-264.313],[-315.8,-264.313],[-277.256,-225.769],[-315.8,-187.225],[-316.988,-187.225]],"c":true}]},{"t":30,"s":[{"i":[[0,21.288],[-21.288,0],[0,0],[0,-21.288],[21.288,0],[0,0]],"o":[[0,-21.288],[0,0],[21.288,0],[0,21.288],[0,0],[-21.288,0]],"v":[[-355.532,-225.769],[-316.988,-264.313],[-151.8,-264.313],[-113.256,-225.769],[-151.8,-187.225],[-316.988,-187.225]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[1,1,1,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":1,"k":[{"i":{"x":0.2,"y":1},"o":{"x":0.8,"y":0},"t":21,"s":[-60,0],"to":[10,0],"ti":[-10,0]},{"t":30,"s":[0,0]}],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":21,"s":[0]},{"t":22,"s":[100]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 2","np":3,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false}],"ip":21,"op":34,"st":15,"bm":0},{"ddd":0,"ind":7,"ty":1,"nm":"Zap In Mask","td":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[800,-250,0],"ix":2},"a":{"a":0,"k":[1000,1000,0],"ix":1},"s":{"a":0,"k":[154.819,50,100],"ix":6}},"ao":0,"sw":2000,"sh":2000,"sc":"#b25151","ip":6,"op":61,"st":0,"bm":0},{"ddd":0,"ind":8,"ty":4,"nm":"Zap In","tt":2,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.712],"y":[1]},"o":{"x":[0.422],"y":[0]},"t":11,"s":[45]},{"t":14,"s":[0]}],"ix":10},"p":{"a":1,"k":[{"i":{"x":0.998,"y":1},"o":{"x":0.326,"y":0},"t":11,"s":[1400,-66,0],"to":[-320,140.333,0],"ti":[90,-388.333,0]},{"t":14,"s":[800,800,0]}],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.001,"y":0},"t":14,"s":[{"i":[[0,0],[11.854,-24.747],[0,0],[-8.245,44.5],[0,0],[0,0],[-6.137,21.277],[-109.472,85.584],[8.971,-44.074],[0,0]],"o":[[17.889,0],[-153.304,171.934],[-27.254,36.782],[0,0],[0,0],[-18.182,0],[44.863,-150.723],[28.39,-29.304],[0,0],[0,0]],"v":[[381.019,-179.626],[398.913,-146.107],[31.329,415.621],[-47.759,383.523],[42.236,108.739],[-79.349,158.739],[-96.863,124.723],[217.472,-397.584],[293.656,-369.786],[218.166,-128.678]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.167,"y":0},"t":16,"s":[{"i":[[-74.337,29.678],[2.769,-23.841],[42.415,-34.984],[4.441,20.529],[18.764,41.261],[30.349,19.261],[-16.137,25.277],[-18.528,32.416],[-7.656,-56.214],[0,0]],"o":[[24.663,-8.322],[-33.231,203.16],[-28.647,6.432],[0,0],[-20.236,-0.739],[-20.651,-11.738],[78.863,-66.723],[18.528,-32.416],[10.344,75.786],[0,0]],"v":[[333.337,-17.678],[373.231,16.841],[43.647,438.568],[-36.441,425.471],[-21.764,194.739],[-203.349,156.738],[-218.863,98.723],[83.472,-237.584],[161.656,-203.786],[110.166,23.322]],"c":true}]},{"i":{"x":0.4,"y":1},"o":{"x":0.167,"y":0},"t":17,"s":[{"i":[[-74.337,29.678],[2.769,-23.841],[46.227,-29.766],[-2.559,47.529],[18.764,41.261],[30.349,19.261],[-16.137,25.277],[-18.528,32.416],[-7.656,-56.214],[0,0]],"o":[[24.663,-8.322],[-33.231,203.16],[-44.647,0.432],[0,0],[-20.236,-0.739],[-20.651,-11.739],[78.863,-66.723],[18.528,-32.416],[10.344,75.786],[0,0]],"v":[[303.337,57.322],[343.231,91.841],[40.647,438.568],[-37.441,402.471],[-39.764,242.739],[-221.349,204.738],[-236.863,146.723],[43.472,-126.584],[121.656,-92.786],[80.166,98.322]],"c":true}]},{"t":19,"s":[{"i":[[0,0],[10.401,-14.029],[0,0],[-8.245,44.5],[0,0],[0,0],[-10.998,13.954],[0,0],[8.972,-44.074],[0,0]],"o":[[17.889,0],[0,0],[-27.254,36.782],[0,0],[0,0],[-18.182,0],[0,0],[28.182,-35.766],[0,0],[0,0]],"v":[[415.337,-136.678],[433.231,-103.159],[43.647,422.569],[-35.441,390.471],[18.236,100.739],[-201.349,100.739],[-218.863,66.723],[167.472,-423.584],[245.656,-389.786],[194.166,-136.678]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":14,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":15,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":16,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":17,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":18,"s":[1,1,1,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":19,"s":[1,0.627451002598,0.184313729405,1]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":20,"s":[1,1,1,1]},{"t":21,"s":[1,0.627451002598,0.184313729405,1]}],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":6,"op":61,"st":6,"bm":0},{"ddd":0,"ind":9,"ty":1,"nm":"Zap Out Mask","td":1,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[800,1774,0],"ix":2},"a":{"a":0,"k":[1000,1000,0],"ix":1},"s":{"a":0,"k":[100,51.6,100],"ix":6}},"ao":0,"sw":2000,"sh":2000,"sc":"#b25151","ip":0,"op":61,"st":0,"bm":0},{"ddd":0,"ind":10,"ty":4,"nm":"Zap Out","tt":2,"sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.999],"y":[1]},"o":{"x":[0.6],"y":[0]},"t":6,"s":[0]},{"t":12,"s":[-12.1]}],"ix":10},"p":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":0,"s":[800,800,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.999,"y":1},"o":{"x":0.6,"y":0},"t":6,"s":[850,702,0],"to":[0,0,0],"ti":[61.667,-551,0]},{"t":12,"s":[328,1762,0]}],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"hasMask":true,"masksProperties":[{"inv":false,"mode":"s","pt":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":0,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[101.138,-61.986],[143.265,-269.065],[-90.054,27.047],[110.668,27.047],[66.549,265.193],[309.001,-61.986]],"c":true}]},{"i":{"x":0.833,"y":1},"o":{"x":0.6,"y":0},"t":6,"s":[{"i":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0],[0,0],[0,0]],"v":[[134.138,-96.986],[152.265,-236.065],[-46.054,-38.953],[184.668,-64.953],[137.549,156.193],[329.001,-122.986]],"c":true}]},{"t":9,"s":[{"i":[[0,0],[-1.641,24.148],[0,0],[0,0],[0,0],[-104.081,129.05]],"o":[[0,0],[-78.547,99.038],[0,0],[0,0],[0,0],[-27.838,4.667]],"v":[[101.138,-61.986],[143.265,-269.065],[-36.054,49.047],[110.668,27.047],[66.549,265.193],[287.001,-97.986]],"c":true}]}],"ix":1},"o":{"a":0,"k":100,"ix":3},"x":{"a":0,"k":0,"ix":4},"nm":"Mask 1"}],"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":0,"s":[{"i":[[0,0],[10.401,-14.029],[0,0],[-8.245,44.5],[0,0],[0,0],[-10.998,13.954],[0,0],[8.972,-44.074],[0,0]],"o":[[17.889,0],[0,0],[-27.254,36.782],[0,0],[0,0],[-18.182,0],[0,0],[28.182,-35.766],[0,0],[0,0]],"v":[[415.337,-136.678],[433.231,-103.159],[43.647,422.569],[-35.441,390.471],[18.236,100.739],[-201.349,100.739],[-218.863,66.723],[167.472,-423.584],[245.656,-389.786],[194.166,-136.678]],"c":true}]},{"i":{"x":0.833,"y":1},"o":{"x":0.6,"y":0},"t":6,"s":[{"i":[[0,0],[10.401,-14.029],[0,0],[-8.245,44.5],[0,0],[0,0],[-10.998,13.954],[0,0],[8.972,-44.074],[0,0]],"o":[[17.889,0],[0,0],[-27.254,36.782],[0,0],[0,0],[-18.182,0],[0,0],[28.182,-35.766],[0,0],[0,0]],"v":[[435.337,-197.678],[453.231,-164.159],[114.647,313.569],[35.559,281.471],[92.236,8.739],[-157.349,34.739],[-174.863,0.722],[176.472,-390.584],[254.656,-356.786],[227.166,-171.678]],"c":true}]},{"t":9,"s":[{"i":[[0,0],[11.854,-24.747],[0,0],[-8.245,44.5],[0,0],[0,0],[-10.998,13.954],[-86.31,83.162],[8.972,-44.074],[0,0]],"o":[[17.889,0],[-153.304,171.933],[-27.254,36.782],[0,0],[0,0],[-18.182,0],[0,0],[28.39,-29.304],[0,0],[0,0]],"v":[[393.337,-172.678],[411.231,-139.159],[43.647,422.569],[-35.441,390.471],[18.236,100.739],[-147.349,122.739],[-164.863,88.723],[167.472,-423.584],[245.656,-389.786],[194.166,-136.678]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":6,"s":[0.4,0.4,0.4,1]},{"t":11,"s":[1,1,1,1]}],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":61,"st":0,"bm":0},{"ddd":0,"ind":11,"ty":4,"nm":"Star Dash Out","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[800,800,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":2,"s":[{"i":[[0,21.291],[-21.288,0],[0,0],[0,-21.286],[21.288,0],[0,0]],"o":[[0,-21.286],[0,0],[21.288,0],[0,21.291],[0,0],[-21.288,0]],"v":[[-399.582,214.731],[-361.038,176.187],[-173.825,176.187],[-135.281,214.731],[-173.825,253.274],[-361.038,253.274]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":7,"s":[{"i":[[0,21.291],[-21.288,0],[0,0],[0,-21.286],[21.288,0],[0,0]],"o":[[0,-21.286],[0,0],[21.288,0],[0,21.291],[0,0],[-21.288,0]],"v":[[-219.582,214.01],[-181.038,175.466],[-173.825,176.187],[-135.281,214.731],[-173.825,253.274],[-181.038,252.554]],"c":true}]},{"t":9,"s":[{"i":[[0,3.754],[-3.753,0],[0,0],[0,-3.753],[3.753,0],[0,0]],"o":[[0,-3.753],[0,0],[3.753,0],[0,3.754],[0,0],[-3.753,0]],"v":[[-184.863,214.307],[-178.067,207.511],[-176.796,207.638],[-170,214.434],[-176.796,221.23],[-178.067,221.103]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.4,0.4,0.4,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":9,"s":[100]},{"t":10,"s":[0]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 4","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":1,"s":[{"i":[[0,0],[0,-21.288],[-21.288,0],[0,0],[0,21.288],[21.288,0]],"o":[[-21.288,0],[0,21.288],[0,0],[21.288,0],[0,-21.288],[0,0]],"v":[[-405.088,-44.063],[-443.632,-5.519],[-405.088,33.025],[-305.975,33.025],[-267.432,-5.519],[-305.975,-44.063]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":6,"s":[{"i":[[0,0],[0,-21.288],[-21.288,0],[0,0],[0,21.288],[21.288,0]],"o":[[-21.288,0],[0,21.288],[0,0],[21.288,0],[0,-21.288],[0,0]],"v":[[-313.088,-43.582],[-351.632,-5.038],[-313.088,33.506],[-305.975,33.025],[-267.432,-5.519],[-305.975,-44.063]],"c":true}]},{"t":8,"s":[{"i":[[0,0],[0,-3.689],[-3.689,0],[0,0],[0,3.689],[3.689,0]],"o":[[-3.689,0],[0,3.689],[0,0],[3.689,0],[0,-3.689],[0,0]],"v":[[-310.148,-11.917],[-316.828,-5.237],[-310.148,1.443],[-308.915,1.359],[-302.236,-5.32],[-308.915,-12]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.4,0.4,0.4,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":8,"s":[100]},{"t":9,"s":[0]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 3","np":3,"cix":2,"bm":0,"ix":2,"mn":"ADBE Vector Group","hd":false},{"ty":"gr","it":[{"ind":0,"ty":"sh","ix":1,"ks":{"a":1,"k":[{"i":{"x":0.4,"y":1},"o":{"x":0.6,"y":0},"t":0,"s":[{"i":[[0,21.288],[-21.288,0],[0,0],[0,-21.288],[21.288,0],[0,0]],"o":[[0,-21.288],[0,0],[21.288,0],[0,21.288],[0,0],[-21.288,0]],"v":[[-355.532,-225.769],[-316.988,-264.313],[-151.8,-264.313],[-113.256,-225.769],[-151.8,-187.225],[-316.988,-187.225]],"c":true}]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":5,"s":[{"i":[[0,21.288],[-21.288,0],[0,0],[0,-21.288],[21.288,0],[0,0]],"o":[[0,-21.288],[0,0],[21.288,0],[0,21.288],[0,0],[-21.288,0]],"v":[[-199.532,-226.495],[-160.988,-265.039],[-151.8,-264.313],[-113.256,-225.769],[-151.8,-187.225],[-160.988,-187.951]],"c":true}]},{"t":7,"s":[{"i":[[0,3.155],[-3.155,0],[0,0],[0,-3.155],[3.155,0],[0,0]],"o":[[0,-3.155],[0,0],[3.155,0],[0,3.155],[0,0],[-3.155,0]],"v":[[-162.788,-226.186],[-157.075,-231.899],[-155.713,-231.791],[-150,-226.078],[-155.713,-220.365],[-157.075,-220.473]],"c":true}]}],"ix":2},"nm":"Path 1","mn":"ADBE Vector Shape - Group","hd":false},{"ty":"fl","c":{"a":0,"k":[0.4,0.4,0.4,1],"ix":4},"o":{"a":0,"k":100,"ix":5},"r":1,"bm":0,"nm":"Fill 1","mn":"ADBE Vector Graphic - Fill","hd":false},{"ty":"tr","p":{"a":0,"k":[0,0],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[100,100],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":7,"s":[100]},{"t":8,"s":[0]}],"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Shape 2","np":3,"cix":2,"bm":0,"ix":3,"mn":"ADBE Vector Group","hd":false}],"ip":0,"op":40,"st":0,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/src/components/Avatar/Avatar.module.scss b/src/components/Avatar/Avatar.module.scss new file mode 100644 index 0000000..d41e87e --- /dev/null +++ b/src/components/Avatar/Avatar.module.scss @@ -0,0 +1,282 @@ +@mixin avatar { + position: relative; + background-color: var(--background-site); + border-radius: 50%; + + img { + border-radius: 50%; + width: 100%; + height: 100%; + object-fit: cover; + vertical-align: top; + } +} + +.verifiedIcon { + position: absolute; + top: 0px; + right: 0px; + width: 15px; + height: 15px; + display: inline-block; + margin: 0px 0px; + background-color: var(--accent-2); + -webkit-mask: url(../../assets/icons/verified.svg) no-repeat 0px / 15px; + mask: url(../../assets/icons/verified.svg) no-repeat 0px / 15px; +} + +@mixin iconBackground { + position: absolute; + right: 0px; + bottom: 0px; + width: 15px; + height: 15px; + background-color: var(--background-site); + border-radius: 50%; +} + +.xxsAvatar { + @include avatar; + width: 28px; + height: 28px; + + .missingBack { + width: 28px; + height: 28px; + } + + .iconBackground { + @include iconBackground; + bottom: -6px; + right: -6px; + } +} + +.xssAvatar { + @include avatar; + width: 26px; + height: 26px; + + .missingBack { + width: 26px; + height: 26px; + } + + .iconBackground { + @include iconBackground; + bottom: -6px; + right: -6px; + } +} + +.xsAvatar { + @include avatar; + width: 32px; + height: 32px; + + .missingBack { + width: 32px; + height: 32px; + } + + .iconBackground { + @include iconBackground; + bottom: -6px; + right: -6px; + } +} + +.vsAvatar { + @include avatar; + width: 42px; + height: 42px; + + .missingBack { + width: 42px; + height: 42px; + } + + .iconBackground { + @include iconBackground; + bottom: 0px; + right: 0px; + } +} + +.smallAvatar { + @include avatar; + width: 48px; + height: 48px; + + .missingBack { + width: 44px; + height: 44px; + } + + .iconBackground { + @include iconBackground; + bottom: 0px; + right: 0px; + } +} + +.midAvatar { + @include avatar; + width: 52px; + height: 52px; + + .missingBack { + width: 52px; + height: 52px; + } + + .iconBackground { + @include iconBackground; + bottom: 0px; + right: 0px; + } +} + +.largeAvatar { + @include avatar; + width: 72px; + height: 72px; + + .missingBack { + width: 72px; + height: 72px; + } + + .iconBackground { + @include iconBackground; + bottom: 3px; + right: 3px; + } +} + +.extraLargeAvatar { + @include avatar; + width: 80px; + height: 80px; + + .missingBack { + width: 80px; + height: 80px; + } + + .iconBackground { + @include iconBackground; + bottom: 4px; + right: 4px; + } +} + +.xxlAvatar { + @include avatar; + width: 142px; + height: 142px; + + .missingBack { + width: 142px; + height: 142px; + } + + .iconBackground { + @include iconBackground; + bottom: 4px; + right: 4px; + } +} + +.missingBack { + background-color: var(--background-site); + border-radius: 50%; +} + +@mixin missing { + display: grid; + place-items: center; + color: var(--missing-avatar-text); + background-color: var(--subtile-devider); + border-radius: 50%; + mask-image: url(../../assets/icons/default_nostrich.svg); + mask-repeat: no-repeat; + -webkit-mask-image: url(../../assets/icons/default_nostrich.svg); + -webkit-mask-repeat: no-repeat; + -webkit-mask-size: contain; + mask-size: contain; +} + +.xxsMissing { + @include missing; + width: 28px; + height: 28px; + font-size: 10px; +} + +.xssMissing { + @include missing; + width: 26px; + height: 26px; + font-size: 10px; +} + +.xsMissing { + @include missing; + width: 32px; + height: 32px; + font-size: 10px; +} + +.vsMissing { + @include missing; + width: 42px; + height: 42px; + font-size: 10px; +} + +.smallMissing { + @include missing; + width: 44px; + height: 44px; + font-size: 12px; +} + +.midMissing { + @include missing; + width: 48px; + height: 48px; + font-size: 16px; +} + +.largeMissing { + @include missing; + width: 68px; + height: 68px; + font-size: 18px; +} + +.extraLargeMissing { + @include missing; + width: 76px; + height: 76px; + font-size: 20px; +} + +.xxlMissing { + @include missing; + width: 142px; + height: 142px; + font-size: 20px; +} + +.highlightBorder { + background: var(--brand-gradient); + padding: 2px; +} + +.cacheFlag { + img { + filter: sepia(100%) saturate(300%) brightness(70%) hue-rotate(300deg) blur(0.1rem); + } +} diff --git a/src/components/Avatar/Avatar.tsx b/src/components/Avatar/Avatar.tsx new file mode 100644 index 0000000..df526b2 --- /dev/null +++ b/src/components/Avatar/Avatar.tsx @@ -0,0 +1,121 @@ +import { Component, createMemo, createSignal, Show } from 'solid-js'; +import defaultAvatar from '../../assets/icons/default_nostrich.svg'; +import { useMediaContext } from '../../contexts/MediaContext'; +import { getMediaUrl } from '../../lib/media'; +import { MediaSize } from '../../types/primal'; + +import styles from './Avatar.module.scss'; + +const Avatar: Component<{ + src: string | undefined, + size?: "xxs" | "xss" | "xs" | "vs" | "sm" | "md" | "lg" | "xl" | "xxl", + verified?: string, + highlightBorder?: boolean, +}> = (props) => { + + const media = useMediaContext(); + + const [isCached, setIsCached] = createSignal(false); + + const selectedSize = props.size || 'sm'; + + const avatarClass = { + xxs: styles.xxsAvatar, + xss: styles.xssAvatar, + xs: styles.xsAvatar, + vs: styles.vsAvatar, + sm: styles.smallAvatar, + md: styles.midAvatar, + lg: styles.largeAvatar, + xl: styles.extraLargeAvatar, + xxl: styles.xxlAvatar, + }; + + const missingClass = { + xxs: styles.xxsMissing, + xss: styles.xssMissing, + xs: styles.xsMissing, + vs: styles.vsMissing, + sm: styles.smallMissing, + md: styles.midMissing, + lg: styles.largeMissing, + xl: styles.extraLargeMissing, + xxl: styles.xxlMissing, + }; + + const imgError = (event: any) => { + const image = event.target; + image.onerror = ""; + image.src = defaultAvatar; + return true; + } + + const highlightClass = () => { + if (props.highlightBorder) { + return styles.highlightBorder; + } + + return ''; + }; + + + const imageSrc = createMemo(() => { + let size: MediaSize = 'm'; + + switch (selectedSize) { + case 'xxs': + case 'xss': + case 'xs': + case 'vs': + case 'sm': + case 'md': + case 'lg': + size = 's'; + break; + default: + size = 'm'; + break; + }; + + const url = media?.actions.getMediaUrl(props.src, size, true); + + setIsCached(!!url); + + return url ?? props.src; + }); + + const notCachedFlag = () => { + const dev = JSON.parse(localStorage.getItem('devMode') || 'false'); + + // @ts-ignore + if (isCached() || !dev) { + return ''; + } + + return styles.cacheFlag; + } + + return ( +
+ +
+
+ } + > +
+ avatar +
+ + +
+
+
+
+ + ) +} + +export default Avatar; diff --git a/src/components/Branding/Branding.module.scss b/src/components/Branding/Branding.module.scss new file mode 100644 index 0000000..2b2aec0 --- /dev/null +++ b/src/components/Branding/Branding.module.scss @@ -0,0 +1,53 @@ +.branding { + display: flex; + align-items: center; + .logo { + background-image: var(--logo); + background-size: contain; + margin-right: 12px; + width: 40px; + height: 40px; + } + + span { + // margin-right: 12px; + font-weight: 700; + font-size: 30px; + line-height: 30px; + color: var(--brand-text); + text-transform: lowercase; + } +} + +.brandingSmall { + .logo { + background-image: var(--logo); + background-size: contain; + margin-right: 0px; + width: 36px; + height: 36px; + } +} + +@media only screen and (max-width: 1300px) { + .branding { + .logo { + margin-right: 0px; + width: 32px; + height: 32px; + } + + span { + display: none; + } + } +} + +.logoLink { + text-decoration: none; + border: none; + border-radius: 0px; + margin: 0px; + padding: 0px; + background: unset; +} diff --git a/src/components/Branding/Branding.tsx b/src/components/Branding/Branding.tsx new file mode 100644 index 0000000..15e24ad --- /dev/null +++ b/src/components/Branding/Branding.tsx @@ -0,0 +1,45 @@ +import { Component, Show } from 'solid-js'; + +import styles from './Branding.module.scss'; +import { useNavigate } from '@solidjs/router'; +import { useIntl } from '@cookbook/solid-intl'; +import { branding } from '../../translations'; + +const Branding: Component<{ small: boolean, isHome?: boolean }> = (props) => { + const navigate = useNavigate(); + const intl = useIntl(); + + const onClick = () => { + if (props.isHome) { + window.scrollTo({ top: 0, left: 0, behavior: 'smooth' }); + return; + } + + navigate('/home'); + } + + return ( + + ) +} + +export default Branding; diff --git a/src/components/Checkbox/Checkbox.module.scss b/src/components/Checkbox/Checkbox.module.scss new file mode 100644 index 0000000..66cf765 --- /dev/null +++ b/src/components/Checkbox/Checkbox.module.scss @@ -0,0 +1,19 @@ +.checkbox { + input { + display: inline-block; + vertical-align: middle; + } + + img { + height: 20px; + margin-right: 12px; + } + + label { + display: inline-block; + font-weight: 400; + font-size: 14px; + line-height: 32px; + color: var(--text-secondary); + } +} diff --git a/src/components/Checkbox/Checkbox.tsx b/src/components/Checkbox/Checkbox.tsx new file mode 100644 index 0000000..22b6427 --- /dev/null +++ b/src/components/Checkbox/Checkbox.tsx @@ -0,0 +1,29 @@ +import { Component, Show } from 'solid-js'; + +import styles from './Checkbox.module.scss'; + +const Checkbox: Component<{ + id: string, + onChange: (e: Event) => void, + checked?: boolean, + label: string, + icon?: string, +}> = (props) => { + + return ( +
+ + + + + +
+ ) +} + +export default Checkbox; diff --git a/src/components/CustomZap/CustomZap.module.scss b/src/components/CustomZap/CustomZap.module.scss new file mode 100644 index 0000000..a5d0ad0 --- /dev/null +++ b/src/components/CustomZap/CustomZap.module.scss @@ -0,0 +1,142 @@ +.customZap { + position: fixed; + width: 420px; + height: 285px; + color: var(--text-secondary); + background-color: var(--background-site); + background: linear-gradient(var(--background-site), var(--background-site)) padding-box, + var(--brand-gradient) border-box; + border: 1px solid transparent; + border-radius: 6px; + + display: flex; + flex-direction: column; + padding: 22px; + + .header { + display: flex; + flex-direction: row; + justify-content: space-between; + + .title { + display: flex; + justify-content: flex-start; + align-items: flex-start; + + .caption { + font-weight: 800; + font-size: 18px; + line-height: 18px; + color: var(--text-secondary); + text-transform: uppercase; + + .amount { + color: var(--text-primary); + } + + .units { + font-weight: 800; + font-size: 12px; + line-height: 20px; + } + } + } + + .close { + border: none; + outline: none; + padding: 0; + margin: 0; + box-shadow: none; + width: 20px; + height: 20px; + display: inline-block; + margin: 0px 0px; + background-color: var(--text-secondary); + -webkit-mask: url(../../assets/icons/close.svg) no-repeat center; + mask: url(../../assets/icons/close.svg) no-repeat center; + + &:hover { + background-color: var(--text-primary); + } + } + } + + .options { + display: grid; + grid-template-columns: 1fr 1fr 1fr; + grid-column-gap: 12px; + grid-row-gap: 16px; + margin-top: 24px; + + .zapOption { + height: 40px; + border-radius: 20px; + border: none; + outline: none; + padding: 0; + margin: 0; + box-shadow: none; + background-color: var(--background-input); + color: var(--text-primary); + text-transform: uppercase; + + &.selected, &:hover { + background-color: var(--text-primary); + color: var(--background-site); + } + + } + } + + .comment { + width: 100%; + height: 32px; + border: 2px solid var(--background-input); + border-radius: 6px; + background-color: var(--background-site); + color: var(--text-secondary); + padding-block: 6px; + padding-inline: 13px; + font-weight: 400; + font-size: 14px; + line-height: 20px; + margin-top: 16px; + + &::placeholder { + color: var(--subtile-devider); + opacity: 1; + font-weight: 400; + font-size: 14px; + line-height: 20px; + } + } + + .action { + width: 100%; + height: 32px; + background: var(--brand-gradient-vertical); + border: none; + outline: none; + padding: 0; + margin: 0; + box-shadow: none; + .caption { + font-weight: 700; + font-size: 14px; + line-height: 20px; + color: white; + } + } + +} + +.zapIcon { + width: 22px; + height: 22px; + display: inline-block; + margin-right: 9px; + background: var(--sidebar-section-icon-gradient); + -webkit-mask: url(../../assets/icons/explore/zaps_hollow.svg) no-repeat 2px 0 / 19px 22px; + mask: url(../../assets/icons/explore/zaps_hollow.svg) no-repeat 2px 0 / 19px 22px; +} diff --git a/src/components/CustomZap/CustomZap.tsx b/src/components/CustomZap/CustomZap.tsx new file mode 100644 index 0000000..2ad9858 --- /dev/null +++ b/src/components/CustomZap/CustomZap.tsx @@ -0,0 +1,156 @@ +import { useIntl } from '@cookbook/solid-intl'; +import { Component, createEffect, createSignal, For } from 'solid-js'; +import { useAccountContext } from '../../contexts/AccountContext'; +import { useSettingsContext } from '../../contexts/SettingsContext'; +import { zapNote } from '../../lib/zap'; +import { userName } from '../../stores/profile'; +import { toastZapFail, zapCustomOption } from '../../translations'; +import { PrimalNote } from '../../types/primal'; +import { debounce } from '../../utils'; +import Modal from '../Modal/Modal'; +import { useToastContext } from '../Toaster/Toaster'; + +import styles from './CustomZap.module.scss'; + +const CustomZap: Component<{ + open?: boolean, + note: PrimalNote, + onConfirm: (amount?: number) => void, + onSuccess: (amount?: number) => void, + onFail: (amount?: number) => void +}> = (props) => { + + const toast = useToastContext(); + const account = useAccountContext(); + const intl = useIntl(); + const settings = useSettingsContext(); + + const [selectedValue, setSelectedValue] = createSignal(settings?.availableZapOptions[0] || 10); + + createEffect(() => { + setSelectedValue(settings?.availableZapOptions[0] || 10) + }); + + const isSelected = (value: number) => value === selectedValue(); + + let comment = ''; + + const setComment = (e: InputEvent) => { + debounce(() => { + const target = e.target as HTMLInputElement; + comment = target.value; + }, 500); + }; + + const truncateNumber = (amount: number) => { + const t = 1000; + + if (amount < t) { + return `${amount}`; + } + + if (amount < Math.pow(t, 2)) { + return (amount % t === 0) ? + `${Math.floor(amount / t)}K` : + intl.formatNumber(amount); + } + + if (amount < Math.pow(t, 3)) { + return (amount % t === 0) ? + `${Math.floor(amount / Math.pow(t, 2))}M` : + intl.formatNumber(amount); + } + + if (amount < Math.pow(t, 4)) { + return (amount % t === 0) ? + `${Math.floor(amount / Math.pow(t, 3))}B` : + intl.formatNumber(amount); + } + + if (amount < Math.pow(t, 5)) { + return (amount % t === 0) ? + `${Math.floor(amount / Math.pow(t, 3))}T` : + intl.formatNumber(amount); + } + + return intl.formatNumber(amount); + }; + + const submit = async () => { + if (account?.hasPublicKey()) { + props.onConfirm(selectedValue()); + const success = await zapNote(props.note, account.publicKey, selectedValue(), comment, account.relays); + + if (success) { + props.onSuccess(selectedValue()); + return; + } + + toast?.sendWarning( + intl.formatMessage(toastZapFail), + ); + + props.onFail(selectedValue()) + } + }; + + return ( + +
+
+
+
+
+ {intl.formatMessage(zapCustomOption,{ + user: userName(props.note.user), + })} + + {truncateNumber(selectedValue())} + sats +
+
+ +
+ +
+ + {(value) => + + } + +
+ + + + + +
+
+ ); +} + +export default CustomZap; diff --git a/src/components/EmbeddedNote/EmbeddedNote.module.scss b/src/components/EmbeddedNote/EmbeddedNote.module.scss new file mode 100644 index 0000000..8de8fbe --- /dev/null +++ b/src/components/EmbeddedNote/EmbeddedNote.module.scss @@ -0,0 +1,113 @@ + +.mentionedNote { + border: solid 1px var(--border-embedded-card); + border-radius: 8px; + background-color: var(--background-embedded_card); + margin-block: 6px; + padding: 18px; + display: grid; + grid-template-columns: 1fr; + grid-template-rows: 32px 1fr; + grid-row-gap: 8px; + text-decoration: none; + color: unset; + line-height: 20px; + + &:hover { + text-decoration: none !important; + } + + .mentionedNoteHeader { + display: flex; + justify-content: flex-start; + align-items: center; + color: var(--text-tertiary-2); + + .postInfo { + display: flex; + justify-content: flex-start; + margin-left: 11px; + + .userInfo { + font-size: 16px; + line-height: 16px; + font-weight: 400; + display: flex; + align-items: center; + width: auto; + .userName { + font-size: 16px; + line-height: 16px; + font-weight: 700; + color: var(--text-primary); + max-width: 150px; + text-overflow: ellipsis; + white-space: nowrap; + overflow: hidden; + } + .verifiedBy { + max-width: 220px; + text-overflow: ellipsis; + white-space: nowrap; + overflow: hidden; + } + } + + .time{ + margin: 0px 2px; + min-width: 40px; + font-size: 14px; + line-height: 16px; + font-weight: 400; + &::before { + content: "|"; + padding: 0px 2px; + } + } + + .contextMenu { + min-width: 5px; + display: inline-block; + text-align: center; + font-weight: bold; + } + } + } +} + +.verifiedIcon { + width: 15px; + height: 15px; + display: inline-block; + margin: 0px 2px; + background-color: var(--text-tertiary-2); + -webkit-mask: url(../../assets/icons/verified.svg) no-repeat 0 / 100%; + mask: url(../../assets/icons/verified.svg) no-repeat 0 / 100%; +} + +@media only screen and (max-width: 720px) { + .mentionedNoteHeader { + width: calc(100vw - 160px); + .postInfo { + width: calc(100vw - 110px); + .userInfo { + max-width: calc(100vw - 100px); + overflow: hidden; + .userName { + max-width: calc(100vw - 180px); + } + } + .verification { + max-width: 220px; + text-overflow: ellipsis; + white-space: nowrap; + overflow: hidden; + + } + } + } +} + +.noteContent { + overflow: hidden; +} diff --git a/src/components/EmbeddedNote/EmbeddedNote.tsx b/src/components/EmbeddedNote/EmbeddedNote.tsx new file mode 100644 index 0000000..d2d6a88 --- /dev/null +++ b/src/components/EmbeddedNote/EmbeddedNote.tsx @@ -0,0 +1,187 @@ +import { useIntl } from '@cookbook/solid-intl'; +import { A } from '@solidjs/router'; +import { nip19 } from 'nostr-tools'; +import { Component, createMemo, JSXElement, Show } from 'solid-js'; +import { useThreadContext } from '../../contexts/ThreadContext'; +import { date } from '../../lib/dates'; +import { parseNote2 } from '../../lib/notes'; +import { trimVerification } from '../../lib/profile'; +import { nip05Verification, userName } from '../../stores/profile'; +import { note as t } from '../../translations'; +import { PrimalNote, PrimalUser } from '../../types/primal'; +import Avatar from '../Avatar/Avatar'; +import { parseNoteLinks, parseNpubLinks } from '../ParsedNote/ParsedNote'; + +import styles from './EmbeddedNote.module.scss'; + +const EmbeddedNote: Component<{ note: PrimalNote, mentionedUsers?: Record, includeEmbeds?: boolean}> = (props) => { + + const threadContext = useThreadContext(); + const intl = useIntl(); + + const noteId = () => nip19.noteEncode(props.note.post.id); + + const navToThread = () => { + threadContext?.actions.setPrimaryNote(props.note); + }; + + const verification = createMemo(() => { + return trimVerification(props.note.user?.nip05); + }); + + const parsedContent = (text: string) => { + const regex = /\#\[([0-9]*)\]/g; + let parsed = text; + + let refs = []; + let match; + + while((match = regex.exec(text)) !== null) { + refs.push(match[1]); + } + + if (refs.length > 0) { + for(let i =0; i < refs.length; i++) { + let r = parseInt(refs[i]); + + const tag = props.note.post.tags[r]; + if ( + tag[0] === 'e' && + props.note.mentionedNotes && + props.note.mentionedNotes[tag[1]] + ) { + const embeded = ( + + {intl.formatMessage( + t.mentionIndication, + { name: userName(props.note.user) }, + )} + + ); + + // @ts-ignore + parsed = parsed.replace(`#[${r}]`, embeded.outerHTML); + } + + if (tag[0] === 'p' && props.mentionedUsers && props.mentionedUsers[tag[1]]) { + const user = props.mentionedUsers[tag[1]]; + + const link = ( + + @{userName(user)} + + ); + + + // @ts-ignore + parsed = parsed.replace(`#[${r}]`, link.outerHTML); + } + } + } + + return parsed; + + }; + + const highlightHashtags = (text: string) => { + const regex = /(?:\s|^)#[^\s!@#$%^&*(),.?":{}|<>]+/ig; + + return text.replace(regex, (token) => { + const [space, term] = token.split('#'); + const embeded = ( + + {space} + #{term} + + ); + + // @ts-ignore + return embeded.outerHTML; + }); + } + + const wrapper = (children: JSXElement) => { + if (props.includeEmbeds) { + return ( +
+ {children} +
+ ); + } + + return ( + navToThread()} + data-event={props.note.post.id} + data-event-bech32={noteId()} + > + {children} + + ); + }; + + return wrapper( + <> +
+ + + + + {userName(props.note.user)} + + } + > + + {verification()[0]} + + + + {nip05Verification(props.note.user)} + + + + + + {date(props.note.post.created_at || 0).label} + + +
+
+
+ + ); +} + +export default EmbeddedNote; diff --git a/src/components/ExploreMenuItem/ExploreMenuItem.module.scss b/src/components/ExploreMenuItem/ExploreMenuItem.module.scss new file mode 100644 index 0000000..a6daa33 --- /dev/null +++ b/src/components/ExploreMenuItem/ExploreMenuItem.module.scss @@ -0,0 +1,170 @@ +.exploreMenuItem { + display: flex; + width: 100%; + height: 100px; + background-color: var(--background-card); + justify-content: space-between; + padding-left: 24px; + margin-bottom: 16px; + border-radius: 4px; + + .itemInfo { + display: flex; + justify-content: flex-start; + align-items: center; + + .itemData { + display: flex; + flex-direction: column; + + .header { + display: flex; + flex-direction: row; + justify-content: flex-start; + align-items: baseline; + + .itemCaption { + font-size: 18px; + line-height: 24px; + font-weight: 700; + color: var(--text-primary); + margin-right: 12px; + } + + .itemStat { + font-size: 24px; + line-height: 28px; + font-weight: 300; + color: var(--text-primary); + + } + } + .footer { + display: flex; + + .itemDescription { + font-size: 12px; + line-height: 16px; + font-weight: 300; + color: var(--text-secondary); + } + } + } + + } + + .itemOptions { + display: flex; + flex-direction: row; + margin: 0px; + + .option { + width: 72px; + height: 92px; + border: none; + margin: 4px; + padding: 0px; + outline: none; + text-decoration: none; + background-color: unset; + display: flex; + flex-direction: column; + align-items: center; + justify-content: center; + cursor: pointer; + + >div { + width: 20px; + height: 20px; + background-color: var(--text-secondary-2); + margin-bottom: 6px; + } + >span { + font-size: 12px; + line-height: 16px; + font-weight: 400; + color: var(--text-secondary-2); + } + + &:hover { + outline: none; + text-decoration: none; + border: solid 1px var(--subtile-devider); + border-radius: 4px; + background-color: var(--background-input); + >div { + background: var(--highlight-gradient); + } + } + } + } +} + +.flameIcon { + width: 20px; + height: 20px; + display: inline-block; + margin: 0px; + -webkit-mask: url(../../assets/icons/explore/flame.svg) no-repeat 0px / 20px; + mask: url(../../assets/icons/explore/flame.svg) no-repeat 0px / 20px; +} + +.zapsIcon { + width: 20px; + height: 20px; + display: inline-block; + margin: 0px; + -webkit-mask: url(../../assets/icons/explore/zaps.svg) no-repeat 0px / 20px; + mask: url(../../assets/icons/explore/zaps.svg) no-repeat 0px / 20px; +} + +.likesIcon { + width: 20px; + height: 20px; + display: inline-block; + margin: 0px; + -webkit-mask: url(../../assets/icons/explore/likes.svg) no-repeat 0px / 20px; + mask: url(../../assets/icons/explore/likes.svg) no-repeat 0px / 20px; +} + +.clockIcon { + width: 20px; + height: 20px; + display: inline-block; + margin: 0px; + -webkit-mask: url(../../assets/icons/explore/clock.svg) no-repeat 0px / 20px; + mask: url(../../assets/icons/explore/clock.svg) no-repeat 0px / 20px; +} + + +@mixin itemIcon { + width: 36px; + height: 36px; + margin-right: 16px; + display: block; + background-color: var(--text-secondary-2); +} + +.followsIcon { + @include itemIcon(); + -webkit-mask: url(../../assets/icons/explore/follows.svg) no-repeat 0px / 36px; + mask: url(../../assets/icons/explore/follows.svg) no-repeat 0px / 36px; +} + +.tribeIcon { + @include itemIcon(); + -webkit-mask: url(../../assets/icons/explore/tribe.svg) no-repeat 0px / 36px; + mask: url(../../assets/icons/explore/tribe.svg) no-repeat 0px / 36px; +} + +.networkIcon { + @include itemIcon(); + -webkit-mask: url(../../assets/icons/explore/network.svg) no-repeat 0px / 36px; + mask: url(../../assets/icons/explore/network.svg) no-repeat 0px / 36px; +} + +.globalIcon { + @include itemIcon(); + -webkit-mask: url(../../assets/icons/explore/global.svg) no-repeat 0px / 36px; + mask: url(../../assets/icons/explore/global.svg) no-repeat 0px / 36px; +} diff --git a/src/components/ExploreMenuItem/ExploreMenuItem.tsx b/src/components/ExploreMenuItem/ExploreMenuItem.tsx new file mode 100644 index 0000000..fbff8e6 --- /dev/null +++ b/src/components/ExploreMenuItem/ExploreMenuItem.tsx @@ -0,0 +1,85 @@ +import { useIntl } from '@cookbook/solid-intl'; +import { A } from '@solidjs/router'; +import type { Component } from 'solid-js'; +import { scopeDescriptors, timeframeDescriptors } from '../../translations'; +import { ScopeDescriptor } from '../../types/primal'; + +import styles from './ExploreMenuItem.module.scss'; + + +const itemInfo: Record = { + follows: { + ...scopeDescriptors.follows, + icon: styles.followsIcon, + }, + tribe: { + ...scopeDescriptors.tribe, + icon: styles.tribeIcon, + }, + network: { + ...scopeDescriptors.network, + icon: styles.networkIcon, + }, + global: { + ...scopeDescriptors.global, + icon: styles.globalIcon, + }, +}; + +const timeframeIcons: Record = { + trending: styles.flameIcon, + mostzapped: styles.zapsIcon, + popular: styles.likesIcon, + latest: styles.clockIcon, +} + +const ExploreMenuItem: Component<{ scope: string, stat: number }> = (props) => { + + const intl = useIntl(); + + const item = () => itemInfo[props.scope]; + + const timeframeOption = (timeframe: string) => { + return ( + +
+ {intl.formatMessage(timeframeDescriptors[timeframe])} +
+ ); + } + + return ( +
+
+
+ +
+
+
+ {intl.formatMessage(item().caption)} +
+
+ {props.stat.toLocaleString()} +
+
+
+
+ {intl.formatMessage(item().description)} +
+
+
+
+
+ {timeframeOption('trending')} + {timeframeOption('mostzapped')} + {timeframeOption('popular')} + {timeframeOption('latest')} +
+
+ ) +} + +export default ExploreMenuItem; diff --git a/src/components/ExploreSidebar/ExploreSidebar.module.scss b/src/components/ExploreSidebar/ExploreSidebar.module.scss new file mode 100644 index 0000000..a1a58fe --- /dev/null +++ b/src/components/ExploreSidebar/ExploreSidebar.module.scss @@ -0,0 +1,43 @@ +.trendingUsers { + display: grid; + grid-template-columns: 1fr 1fr 1fr 1fr; + + .user { + width: 75px; + height: 85px; + display: flex; + flex-direction: column; + align-items: center; + justify-content: center; + text-decoration: none; + opacity: 0.7; + transition: opacity 0.5s; + + &:hover { + opacity: 1; + transition: opacity 0.5s; + } + + .name { + text-overflow: ellipsis; + white-space: nowrap; + overflow: hidden; + text-align: center; + margin-top: 10px; + width: 72px; + font-size: 12px; + line-height: 12px; + font-weight: 400; + color: var(--text-primary); + } + } +} + +.trendingUsersCaption { + margin-bottom: 24px; + font-weight: 800; + font-size: 18px; + line-height: 20px; + text-transform: uppercase; + color: var(--text-secondary); +} diff --git a/src/components/ExploreSidebar/ExploreSidebar.tsx b/src/components/ExploreSidebar/ExploreSidebar.tsx new file mode 100644 index 0000000..c835f51 --- /dev/null +++ b/src/components/ExploreSidebar/ExploreSidebar.tsx @@ -0,0 +1,144 @@ +import { A, useResolvedPath } from '@solidjs/router'; +import { Component, createEffect, createMemo, For, onCleanup } from 'solid-js'; +import { createStore } from 'solid-js/store'; +import { Kind } from '../../constants'; +import { APP_ID } from '../../App'; +import { getExploreFeed } from '../../lib/feed'; +import { isConnected, refreshSocketListeners, removeSocketListeners, socket } from '../../sockets'; +import { sortingPlan, convertToNotes } from '../../stores/note'; +import { convertToUser, emptyUser, truncateNpub } from '../../stores/profile'; +import { FeedPage, NostrEOSE, NostrEvent, NostrEventContent, NostrUserContent, PrimalNote, PrimalUser } from '../../types/primal'; +import Avatar from '../Avatar/Avatar'; + +import styles from './ExploreSidebar.module.scss'; +import { useIntl } from '@cookbook/solid-intl'; +import { getTrendingUsers } from '../../lib/profile'; +import { hexToNpub } from '../../lib/keys'; +import { exploreSidebarCaption } from '../../translations'; + +const ExploreSidebar: Component = () => { + + const intl = useIntl(); + + const [store, setStore] = createStore<{ users: Record, scores: Record }>({ + users: {}, + scores: {}, + }); + + const [trendingUsers, setTrendingUsers] = createStore([]); + + const authorName = (user: PrimalUser) => { + return user.displayName || + user.name || + truncateNpub(user.npub); + } + +// ACTIONS ------------------------------------- + + const processUsers = (type: string, content: NostrEventContent | undefined) => { + + if (type === 'EOSE') { + const sortedKeys = Object.keys(store.scores).sort( + (a, b) => store.scores[b] - store.scores[a]); + + const users = sortedKeys.map(key => { + if (!store.users[key]) { + return emptyUser(key); + } + + return convertToUser(store.users[key]); + }); + + setTrendingUsers(() => [...users]); + return; + } + + if (type === 'EVENT') { + if (content && content.kind === Kind.Metadata) { + setStore('users', (users) => ({ ...users, [content.pubkey]: content})); + return; + } + if (content && content.kind === Kind.UserScore) { + const scores = JSON.parse(content.content); + + setStore('scores', () => ({ ...scores })); + return; + } + } + }; + + +// SOCKET HANDLERS ------------------------------ + + const onSocketClose = (closeEvent: CloseEvent) => { + const webSocket = closeEvent.target as WebSocket; + + webSocket.removeEventListener('message', onMessage); + webSocket.removeEventListener('close', onSocketClose); + }; + + const onMessage = (event: MessageEvent) => { + const message: NostrEvent | NostrEOSE = JSON.parse(event.data); + + const [type, subId, content] = message; + + if (subId === `explore_sidebar_${APP_ID}`) { + processUsers(type, content); + return; + } + }; + +// EFFECTS -------------------------------------- + + onCleanup(() => { + removeSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + }); + + + createEffect(() => { + if (isConnected()) { + refreshSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + + setStore(() => ({ + users: {}, + scores: {}, + })); + + getTrendingUsers(`explore_sidebar_${APP_ID}`); + } + }); + +// RENDER --------------------------------------- + + return ( + <> +
+ {intl.formatMessage(exploreSidebarCaption)} +
+
+ + { + user => ( + + +
{authorName(user)}
+
+ ) + } +
+
+ + ) +} + +export default ExploreSidebar; diff --git a/src/components/FeedSelect/FeedSelect.tsx b/src/components/FeedSelect/FeedSelect.tsx new file mode 100644 index 0000000..f090925 --- /dev/null +++ b/src/components/FeedSelect/FeedSelect.tsx @@ -0,0 +1,91 @@ +import { Component } from 'solid-js'; +import { useHomeContext } from '../../contexts/HomeContext'; +import { useSettingsContext } from '../../contexts/SettingsContext'; +import { FeedOption } from '../../types/primal'; +import SelectBox from '../SelectBox/SelectBox'; + +const FeedSelect: Component<{ isPhone?: boolean}> = (props) => { + + const home = useHomeContext(); + const settings = useSettingsContext(); + + const selectFeed = (option: FeedOption) => { + const hex = option.value; + const selector = document.getElementById('defocus'); + + selector?.focus(); + selector?.blur(); + + if (hex) { + const feed = settings?.availableFeeds.find(p => p.hex === hex); + + if (hex !== initialValue()?.value) { + home?.actions.clearNotes(); + home?.actions.selectFeed(feed); + } + return; + } + + }; + + const isSelected = (option: FeedOption) => { + const selected = home?.selectedFeed; + + if (selected?.hex) { + return selected.hex === option.value; + } + + return false; + } + + const options:() => FeedOption[] = () => { + if (settings?.availableFeeds === undefined) { + return []; + } + + return settings.availableFeeds.map(feed => { + return ({ + label: feed.name, + value: feed.hex, + }); + }); + }; + + const initialValue = () => { + const selected = home?.selectedFeed; + + if (!selected) { + return { + label: '', + value: undefined, + }; + } + + const feed = settings?.availableFeeds.find(f => + f.hex === selected.hex + ); + + if (feed) { + const [scope, timeframe] = feed.hex?.split(';') || []; + + const value = scope && timeframe ? `${scope};${timeframe}` : feed.hex; + + return { + label: feed.name, + value, + }; + } + } + + return ( + + ); +} + +export default FeedSelect; diff --git a/src/components/FeedSorter/FeedSorter.module.scss b/src/components/FeedSorter/FeedSorter.module.scss new file mode 100644 index 0000000..5fb7441 --- /dev/null +++ b/src/components/FeedSorter/FeedSorter.module.scss @@ -0,0 +1,148 @@ +@mixin controlButton { + margin: 0; + padding: 0; + border: none; + background-color: unset; + + opacity: 0.7; + transition: opacity 0.4s; + + &:hover { + opacity: 1; + transition: opacity 0.4s; + } + + &:focus { + box-shadow: none; + } +} + +.feedItem { + width: 100%; + height: 40px; + background-color: var(--background-site); + color: var(--text-tertiary); + font-size: 16px; + line-height: 14px; + font-weight: 400; + padding-inline: 12px; + margin-block: 8px; + + display: flex; + flex-direction: row; + align-items: center; + justify-content: flex-start; + cursor: grab; + + .sortControls { + margin-right: 5px; + .sortButton { + @include controlButton(); + width: 16px; + height: 16px; + font-size: 16px; + line-height: 16px; + } + } + + .manageControls { + display: flex; + .mngButton { + @include controlButton(); + width: 32px; + height: 32px; + } + + .deleteButton { + width: 16px; + height: 16px; + display: inline-block; + margin: 0px 0px; + background-color: var(--text-secondary); + -webkit-mask: url(../../assets/icons/delete.svg) no-repeat 0px / 16px; + mask: url(../../assets/icons/delete.svg) no-repeat 0px / 16px; + } + + .editButton { + width: 16px; + height: 16px; + display: inline-block; + margin: 0px 0px; + background-color: var(--text-secondary); + -webkit-mask: url(../../assets/icons/edit.svg) no-repeat 0px / 16px; + mask: url(../../assets/icons/edit.svg) no-repeat 0px / 16px; + } + } + + .feedEdit { + position: relative; + width: 100%; + .feedNameInput { + height: 32px; + background-color: var(--background-site); + color: var(--text-tertiary); + font-size: 16px; + line-height: 14px; + font-weight: 400; + margin: 0; + } + .feedEditControl { + position: absolute; + top: 2px; + right: 0; + height: 32px; + display: flex; + justify-content: flex-end; + align-items: center; + + >button { + @include controlButton(); + width: 32px; + height: 32px; + } + } + + } +} + +.dragIcon { + width: 20px; + height: 20px; + display: inline-block; + margin: 0px 0px; + background-color: var(--text-secondary); + -webkit-mask: url(../../assets/icons/drag_handle.svg) no-repeat center; + mask: url(../../assets/icons/drag_handle.svg) no-repeat center; +} + +.closeIcon { + width: 20px; + height: 20px; + display: inline-block; + margin: 0px 0px; + background-color: var(--text-secondary); + -webkit-mask: url(../../assets/icons/close.svg) no-repeat center; + mask: url(../../assets/icons/close.svg) no-repeat center; +} + +.checkIcon { + width: 20px; + height: 20px; + display: inline-block; + margin: 0px 0px; + background-color: var(--text-secondary); + -webkit-mask: url(../../assets/icons/check.svg) no-repeat 0 / 100%; + mask: url(../../assets/icons/check.svg) no-repeat 0 / 100%; +} + +.draggedBefore { + border-top: solid 1px var(--subtile-devider); +} + +.draggedAfter { + border-bottom: solid 1px var(--subtile-devider); +} + +.draggedItem { + opacity: 0.6; +} diff --git a/src/components/FeedSorter/FeedSorter.tsx b/src/components/FeedSorter/FeedSorter.tsx new file mode 100644 index 0000000..7be3269 --- /dev/null +++ b/src/components/FeedSorter/FeedSorter.tsx @@ -0,0 +1,179 @@ +import { Component, createEffect, createSignal, For, Show } from 'solid-js'; +import { useAccountContext } from '../../contexts/AccountContext'; +import { useSettingsContext } from '../../contexts/SettingsContext'; +import { PrimalFeed } from '../../types/primal'; + +import styles from './FeedSorter.module.scss'; + + +const FeedSorter: Component = () => { + + let sorter: any; + + const settings = useSettingsContext(); + const account = useAccountContext(); + + const [editMode, setEditMode] = createSignal(''); + + const [newName, setNewName] = createSignal(''); + + const availableFeeds = () => { + return settings?.availableFeeds || []; + }; + + const removeFeed = (feed: PrimalFeed) => { + settings?.actions.removeAvailableFeed(feed); + }; + + const editFeed = (feed: PrimalFeed) => { + setEditMode(() => feed.hex || ''); + setNewName(() => feed.name); + const input = document.getElementById(`input_${feed.hex}`); + input && input.focus(); + }; + + const updateFeedName = (feed: PrimalFeed) => { + settings?.actions.renameAvailableFeed(feed, newName()); + setEditMode(''); + } + + const sortList = (target: any) => { + // Get all items + let items = target.getElementsByClassName(styles.feedItem); + // init current item + let current: any = null; + + // (Make items draggable and sortable + for (let i of items) { + i.draggable = true; + + i.ondragstart = (e: DragEvent) => { + current = i; + for (let it of items) { + if (it === current) { + it.classList.add(styles.draggedItem); + } + } + }; + + i.ondragenter = (e: DragEvent) => { + const oldIndex = current.getAttribute('data-index'); + const newIndex = i.getAttribute('data-index'); + + if (oldIndex > newIndex) { + i.classList.add(styles.draggedBefore); + i.classList.remove(styles.draggedAfter); + } + if (oldIndex < newIndex) { + i.classList.add(styles.draggedAfter); + i.classList.remove(styles.draggedBefore); + } + }; + + i.ondragleave = () => { + i.classList.remove(styles.draggedBefore); + i.classList.remove(styles.draggedAfter); + } + + i.ondragend = () => { for (let it of items) { + it.classList.remove(styles.draggedItem); + i.classList.remove(styles.draggedBefore); + i.classList.remove(styles.draggedAfter); + }}; + + // Prevent default "drop", so we can do our own + i.ondragover = (e: DragEvent) => e.preventDefault(); + + i.ondrop = (e: DragEvent) => { + e.preventDefault(); + if (i != current) { + const oldIndex = current.getAttribute('data-index'); + const newIndex = i.getAttribute('data-index'); + + settings?.actions.moveAvailableFeed(oldIndex, newIndex); + + for (let it of items) { + it.classList.remove(styles.draggedBefore); + it.classList.remove(styles.draggedBefore); + it.classList.remove(styles.draggeditem); + } + } + }; + } + } + + createEffect(() => { + if (sorter && availableFeeds().length > 0) { + sortList(sorter); + } + }); + + return ( +
+ 0}> + + {(feed, index) => ( +
+ + +
+
+
+
+ + +
+
+
{feed.name}
+ + } + > +
+ setNewName(() => e.target?.value)} + onKeyUp={(e: KeyboardEvent) => { + if (e.code === 'Enter') { + updateFeedName(feed); + } + + if (e.code === 'Escape') { + setEditMode(''); + } + }} + /> +
+ + +
+
+
+
+ )} +
+
+
+ ) +} + +export default FeedSorter; diff --git a/src/components/FloatingNewPostButton/FloatingNewPostButton.module.scss b/src/components/FloatingNewPostButton/FloatingNewPostButton.module.scss new file mode 100644 index 0000000..619176d --- /dev/null +++ b/src/components/FloatingNewPostButton/FloatingNewPostButton.module.scss @@ -0,0 +1,34 @@ +.newPostButton { + width: 48px; + height: 48px; + padding: 0px; + background: var(--brand-gradient); + border-radius: 50%; +} + +.postIcon { + display: inline-block; + width: 24px; + height: 24px; + vertical-align: middle; + margin-top: -3px; + + background-color: var(--missing-avatar-text); + -webkit-mask: url(../../assets/icons/post.svg) no-repeat center; + mask: url(../../assets/icons/post.svg) no-repeat center; +} + + +@media only screen and (max-width: 1300px) { + .newPostButton { + width: 32px; + height: 32px; + } + + .postIcon { + width: 16px; + height: 16px; + -webkit-mask: url(../../assets/icons/post.svg) no-repeat 0px / 16px; + mask: url(../../assets/icons/post.svg) no-repeat 0px / 16px; + } +} diff --git a/src/components/FloatingNewPostButton/FloatingNewPostButton.tsx b/src/components/FloatingNewPostButton/FloatingNewPostButton.tsx new file mode 100644 index 0000000..5e41ed2 --- /dev/null +++ b/src/components/FloatingNewPostButton/FloatingNewPostButton.tsx @@ -0,0 +1,21 @@ +import { useAccountContext } from "../../contexts/AccountContext"; + +import styles from "./FloatingNewPostButton.module.scss"; + +export default function FloatingNewPostButton() { + const account = useAccountContext(); + + const showNewNoteForm = () => { + account?.actions?.showNewNoteForm(); + }; + + + return ( + + ) +} diff --git a/src/components/FollowButton/FollowButton.module.scss b/src/components/FollowButton/FollowButton.module.scss new file mode 100644 index 0000000..e351027 --- /dev/null +++ b/src/components/FollowButton/FollowButton.module.scss @@ -0,0 +1,57 @@ + + @mixin followButton { + grid-area: follow; + align-items: center; + display: flex; + align-items: center; + button { + background: var(--brand-gradient-vertical); + border-radius: 6px; + padding: 0px; + font-size: 12px; + line-height: 16px; + font-weight: 700; + margin: 0px; + } + } + + .primaryButton { + width: 90px; + height: 40px; + } + + .follow { + @include followButton; + button { + border: none; + } + } + + .unfollow { + @include followButton; + button { + color: var(--text-secondary); + background-color: var(--background-card); + background: linear-gradient(var(--background-card), var(--background-card)) padding-box, + var(--brand-gradient) border-box; + border: 1px solid transparent; + } + } + + .large { + button { + width: 90px; + height: 40px; + margin: 0px 8px; + font-size: 16px; + line-height: 20px; + } + + } + + .small { + button { + width: 64px; + height: 40px; + } + } diff --git a/src/components/FollowButton/FollowButton.tsx b/src/components/FollowButton/FollowButton.tsx new file mode 100644 index 0000000..3573b28 --- /dev/null +++ b/src/components/FollowButton/FollowButton.tsx @@ -0,0 +1,57 @@ +import { useIntl } from '@cookbook/solid-intl'; +import { Component, Show } from 'solid-js'; +import { useAccountContext } from '../../contexts/AccountContext'; +import { account as t } from '../../translations'; +import { PrimalUser } from '../../types/primal'; +import { useToastContext } from '../Toaster/Toaster'; + +import styles from './FollowButton.module.scss'; + + +const FollowButton: Component<{ person: PrimalUser | undefined, large?: boolean }> = (props) => { + + const toast = useToastContext() + const account = useAccountContext(); + const intl = useIntl(); + + const isFollowed = () => { + return props.person && + account?.publicKey && + account?.following.includes(props.person.pubkey); + } + + const onFollow = (e: MouseEvent) => { + e.preventDefault(); + if (!account || !account.hasPublicKey() || !props.person) { + toast?.sendWarning(intl.formatMessage(t.needToLogin)) + return; + } + + const action = isFollowed() ? + account.actions.removeFollow : + account.actions.addFollow; + + action(props.person.pubkey); + } + + const klass = () => { + return `${isFollowed() ? styles.unfollow : styles.follow} ${props.large ? styles.large : styles.small}`; + } + + return ( + +
+ +
+
+ ) +} + +export default FollowButton; diff --git a/src/components/HomeHeader/HomeHeader.module.scss b/src/components/HomeHeader/HomeHeader.module.scss new file mode 100644 index 0000000..14f4e0a --- /dev/null +++ b/src/components/HomeHeader/HomeHeader.module.scss @@ -0,0 +1,168 @@ +.fullHeader { + display: grid; + height: 120px; + padding-top: 26px; +} + +.smallHeader { + display: grid; + height: 32px; + grid-template-rows: 1fr 4px; +} + +.smallHeaderMain { + display: flex; + flex-direction: row; + justify-content: stretch; + align-items: flex-end; + background-color: var(--background-site); +} + +.smallHeaderBottomBorder { + width: 100%; + height: 4px; + display: flex; + justify-content: space-between; + // background-color: red; + + .rightCorner { + display: inline-block; + width: 4px; + height: 4px; + + background-color: var(--background-site); + -webkit-mask: url(../../assets/icons/corner_right.svg) no-repeat center; + mask: url(../../assets/icons/corner_right.svg) no-repeat center; + } + .leftCorner { + display: inline-block; + width: 4px; + height: 4px; + + background-color: var(--background-site); + -webkit-mask: url(../../assets/icons/corner_left.svg) no-repeat center; + mask: url(../../assets/icons/corner_left.svg) no-repeat center; + } +} + +.smallLeft { + width: 100%; + flex-grow: 20; + flex-shrink: 1; + visibility: hidden; +} + +.smallRight { + max-width: 50%; + flex-grow: 1; + margin-bottom: 4px; + // margin: auto; +} + +.fixedSelector { + position:fixed; + top: 0px; + width: 640px; + height: 48px; + // align-items: center; + // background-color: var(--background-site); + z-index: var(--z-index-header); + transition: 0.3s; + transform: translateY(0px); + + .smallHeaderMain { + align-items: center; + } + + .smallLeft { + visibility: visible; + } + + .smallRight { + margin-bottom: 0; + } +} + +.hiddenSelector { + transition: 0.3s; + transform: translateY(-48px); +} + +.instaHide { + top: -48px; +} + +.callToAction { + display: grid; + height: 72px; + grid-template-columns: 72px 1fr; + grid-column-gap: 17px; + align-items: center; + background-color: unset; + margin: 0px; + padding: 0px; + border: none; + outline: none; + + p { + font-size: 34px; + line-height: 34px; + padding: 0px; + margin: 0px; + } + + .border { + height: 36px; + padding: 1px; + background: var(--brand-gradient); + border-radius: 6px; + margin-left: 10px + } + + .input { + height: 34px; + font-size: 18px; + line-height: 20px; + margin: 0px; + border-radius: 6px; + border: none; + color: var(--text-tertiary); + background-color: var(--background-site); + display: flex; + align-items: center; + padding-left: 12px; + } +} + +.welcomeMessage { + display: grid; + align-content: center; + height: 72px; + font-weight: 300; + font-size: 32px; + line-height: 34px; + color: var(--brand-text); + text-transform: lowercase; +} + +.welcomeMessageSmall { + display: grid; + align-content: center; + font-weight: 300; + font-size: 24px; + line-height: 28px; + color: var(--brand-text); + text-transform: lowercase; +} + +@media only screen and (max-width: 720px) { + .fullHeader { + width: 100%; + } + .smallHeader { + width: 100%; + } + .fixedSelector { + width: 100%; + } +} diff --git a/src/components/HomeHeader/HomeHeader.tsx b/src/components/HomeHeader/HomeHeader.tsx new file mode 100644 index 0000000..178a571 --- /dev/null +++ b/src/components/HomeHeader/HomeHeader.tsx @@ -0,0 +1,131 @@ +import { Component, onCleanup, onMount, Show } from 'solid-js'; +import Avatar from '../Avatar/Avatar'; + +import styles from './HomeHeader.module.scss'; +import FeedSelect from '../FeedSelect/FeedSelect'; +import { useAccountContext } from '../../contexts/AccountContext'; +import SmallCallToAction from '../SmallCallToAction/SmallCallToAction'; +import { useHomeContext } from '../../contexts/HomeContext'; +import { useIntl } from '@cookbook/solid-intl'; +import { useSettingsContext } from '../../contexts/SettingsContext'; +import { placeholders as t } from '../../translations'; + +const HomeHeader: Component = () => { + + const account = useAccountContext(); + const home = useHomeContext(); + const settings = useSettingsContext(); + const intl = useIntl(); + + let lastScrollTop = document.body.scrollTop || document.documentElement.scrollTop; + + const onScroll = () => { + const scrollTop = document.body.scrollTop || document.documentElement.scrollTop; + const smallHeader = document.getElementById('small_header'); + const border = document.getElementById('small_bottom_border'); + + home?.actions.updateScrollTop(scrollTop); + + const isScrollingDown = scrollTop > lastScrollTop; + lastScrollTop = scrollTop; + + if (scrollTop < 117) { + if (border) { + border.style.display = 'none'; + } + smallHeader?.classList.remove(styles.hiddenSelector); + smallHeader?.classList.remove(styles.fixedSelector); + return; + } + + if (lastScrollTop < 117) { + smallHeader?.classList.add(styles.instaHide); + return; + } + + if (border) { + border.style.display = 'flex'; + } + + smallHeader?.classList.remove(styles.instaHide); + + if (!isScrollingDown) { + smallHeader?.classList.add(styles.fixedSelector); + smallHeader?.classList.remove(styles.hiddenSelector); + return; + } + + smallHeader?.classList.add(styles.hiddenSelector); + } + + const onShowNewNoteinput = () => { + account?.actions?.showNewNoteForm(); + }; + + onMount(() => { + window.addEventListener('scroll', onScroll); + }); + + onCleanup(() => { + window.removeEventListener('scroll', onScroll); + }); + + const activeUser = () => account?.activeUser; + + return ( +
+ + {intl.formatMessage(t.guestUserGreeting)} +
} + > + + + +
+
+ +
+ {intl.formatMessage(t.welcomeMessage)} +
+
} + > +
+ +
+ + + 0 && home?.selectedFeed}> +
+ +
+
+
+
+
+
+
+ + + ); +} + +export default HomeHeader; diff --git a/src/components/HomeHeaderPhone/HomeHeaderPhone.module.scss b/src/components/HomeHeaderPhone/HomeHeaderPhone.module.scss new file mode 100644 index 0000000..ed87ab3 --- /dev/null +++ b/src/components/HomeHeaderPhone/HomeHeaderPhone.module.scss @@ -0,0 +1,109 @@ +.fullHeader { + display: grid; + width: 100%; + grid-template-columns: 1fr; + grid-template-rows: 48px 4px; + grid-template-areas: "phone_header" "phone_border"; +} +.phoneHeader { + grid-area: phone_header; + display: flex; + flex-direction: row; + position: relative; + width: 100%; + height: 48px; + align-items: center; + justify-content: space-between; + padding-inline: 16px; + padding-top: 4px; +} + +.logo { + width: 36px; + margin-right: 16px; +} + + +.smallHeader { + display: grid; + height: 32px; + grid-template-rows: 1fr 4px; +} + +.smallHeaderMain { + display: flex; + flex-direction: row; + justify-content: stretch; + align-items: flex-end; + background-color: var(--background-site); +} + +.smallHeaderBottomBorder { + grid-area: phone_border; + width: 100%; + height: 4px; + display: flex; + justify-content: space-between; + // background-color: red; + + .rightCorner { + display: inline-block; + width: 4px; + height: 4px; + + background-color: var(--background-site); + -webkit-mask: url(../../assets/icons/corner_right.svg) no-repeat center; + mask: url(../../assets/icons/corner_right.svg) no-repeat center; + } + .leftCorner { + display: inline-block; + width: 4px; + height: 4px; + + background-color: var(--background-site); + -webkit-mask: url(../../assets/icons/corner_left.svg) no-repeat center; + mask: url(../../assets/icons/corner_left.svg) no-repeat center; + } +} + +.smallLeft { + width: 100%; + flex-grow: 20; + flex-shrink: 1; + visibility: hidden; +} + +.smallRight { + max-width: 50%; + flex-grow: 1; + // margin: auto; +} + +.fixedSelector { + position:fixed; + top: 0px; + width: 100%; + height: 48px; + // align-items: center; + background-color: var(--background-site); + z-index: var(--z-index-header); + transition: 0.3s; + transform: translateY(0px); + + .smallHeaderMain { + align-items: center; + } + + .smallLeft { + visibility: visible; + } +} + +.hiddenSelector { + transition: 0.3s; + transform: translateY(-48px); +} + +.instaHide { + top: -48px; +} diff --git a/src/components/HomeHeaderPhone/HomeHeaderPhone.tsx b/src/components/HomeHeaderPhone/HomeHeaderPhone.tsx new file mode 100644 index 0000000..ba1a513 --- /dev/null +++ b/src/components/HomeHeaderPhone/HomeHeaderPhone.tsx @@ -0,0 +1,81 @@ +import { Component, onCleanup, onMount, Show } from 'solid-js'; + +import styles from './HomeHeaderPhone.module.scss'; +import FeedSelect from '../FeedSelect/FeedSelect'; +import Branding from '../Branding/Branding'; +import { useHomeContext } from '../../contexts/HomeContext'; + +const HomeHeaderPhone: Component = () => { + + const home = useHomeContext(); + + let lastScrollTop = document.body.scrollTop || document.documentElement.scrollTop; + + const onScroll = () => { + const scrollTop = document.body.scrollTop || document.documentElement.scrollTop; + const smallHeader = document.getElementById('phone_header'); + const border = document.getElementById('small_bottom_border'); + + home?.actions?.updateScrollTop(scrollTop); + + const isScrollingDown = scrollTop > lastScrollTop; + lastScrollTop = scrollTop; + + if (scrollTop < 117) { + if (border) { + border.style.display = 'none'; + } + smallHeader?.classList.remove(styles.hiddenSelector); + smallHeader?.classList.remove(styles.fixedSelector); + return; + } + + if (lastScrollTop < 117) { + smallHeader?.classList.add(styles.instaHide); + return; + } + + if (border) { + border.style.display = 'flex'; + } + smallHeader?.classList.remove(styles.instaHide); + + if (!isScrollingDown) { + smallHeader?.classList.add(styles.fixedSelector); + smallHeader?.classList.remove(styles.hiddenSelector); + return; + } + + smallHeader?.classList.add(styles.hiddenSelector); + } + + onMount(() => { + window.addEventListener('scroll', onScroll); + }); + + onCleanup(() => { + window.removeEventListener('scroll', onScroll); + }); + + return ( +
+
+
+ +
+ + + +
+
+
+
+
+
+ ); +} + +export default HomeHeaderPhone; diff --git a/src/components/HomeSidebar/HomeSidebar.module.scss b/src/components/HomeSidebar/HomeSidebar.module.scss new file mode 100644 index 0000000..c4c8346 --- /dev/null +++ b/src/components/HomeSidebar/HomeSidebar.module.scss @@ -0,0 +1,59 @@ +@mixin heading { + position: -webkit-sticky; + position: sticky; + top: 0px; + width: 100%; + height: 44px; + // background-color: var(--background-site); + background: var(--fade-gradient-vertical); + z-index: 5; + padding-bottom: 22px; + display:flex; + flex-direction: row; + align-items: center; + justify-content: flex-start; + font-size: 18px; + font-weight: 800; + line-height: 22px; + color: var(--text-secondary-2); + text-transform: uppercase; + >div{ + display: flex; + height: 22px; + >span { + color: var(--text-tertiary-2); + text-transform: lowercase; + margin-left: 6px; + } + } +} + +.flameIcon { + width: 22px; + height: 22px; + display: inline-block; + margin-right: 9px; + background: var(--sidebar-section-icon-gradient); + -webkit-mask: url(../../assets/icons/explore/flame.svg) no-repeat 1px 0 / 20px 22px; + mask: url(../../assets/icons/explore/flame.svg) no-repeat 1px 0 / 20px 22px; +} + +.zapIcon { + width: 22px; + height: 22px; + display: inline-block; + margin-right: 9px; + background: var(--sidebar-section-icon-gradient); + -webkit-mask: url(../../assets/icons/explore/zaps_hollow.svg) no-repeat 2px 0 / 19px 22px; + mask: url(../../assets/icons/explore/zaps_hollow.svg) no-repeat 2px 0 / 19px 22px; +} + +.headingTrending { + @include heading(); +} + +.headingZapped { + @include heading(); + margin-top: 34px; + z-index: 10px; +} diff --git a/src/components/HomeSidebar/HomeSidebar.tsx b/src/components/HomeSidebar/HomeSidebar.tsx new file mode 100644 index 0000000..1f21151 --- /dev/null +++ b/src/components/HomeSidebar/HomeSidebar.tsx @@ -0,0 +1,208 @@ +import { Component, createEffect, createSignal, For, onCleanup } from 'solid-js'; +import { createStore } from 'solid-js/store'; +import { APP_ID } from '../../App'; +import { getMostZapped4h, getTrending24h } from '../../lib/feed'; +import { humanizeNumber } from '../../lib/stats'; +import { convertToNotes, sortingPlan } from '../../stores/note'; +import { Kind } from '../../constants'; +import { + isConnected, + refreshSocketListeners, + removeSocketListeners, + socket +} from '../../sockets'; +import { + FeedPage, + NostrEOSE, + NostrEvent, + NostrEventContent, + NostrMentionContent, + PrimalNote +} from '../../types/primal'; + +import styles from './HomeSidebar.module.scss'; +import SmallNote from '../SmallNote/SmallNote'; +import { useIntl } from '@cookbook/solid-intl'; +import { hourNarrow } from '../../formats'; + +const [init, setInit] = createSignal(false); + +const [data, setData] = createStore>({ + trending: { + messages: [], + users: {}, + postStats: {}, + notes: [], + mentions: {}, + }, + mostzapped: { + messages: [], + users: {}, + postStats: {}, + notes: [], + mentions: {}, + }, +}); + +const HomeSidebar: Component = () => { + + const intl = useIntl(); + + onCleanup(() => { + removeSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + }); + + + createEffect(() => { + if (isConnected() && !init()) { + refreshSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + + setData(() => ({ + trending: { + messages: [], + users: {}, + postStats: {}, + notes: [], + mentions: {}, + }, + mostzapped: { + messages: [], + users: {}, + postStats: {}, + notes: [], + mentions: {}, + }, + })); + + getTrending24h(`sidebar_trending_${APP_ID}`); + getMostZapped4h(`sidebar_zapped_${APP_ID}`); + } + }); + + const processNotes = (type: string, key: string, content: NostrEventContent | undefined) => { + + const sort = sortingPlan(key); + + if (type === 'EOSE') { + const newPosts = sort(convertToNotes({ + users: data[key].users, + messages: data[key].messages, + postStats: data[key].postStats, + mentions: data[key].mentions, + })); + + setData(key, 'notes', () => [ ...newPosts ]); + + setInit(true); + return; + } + + if (type === 'EVENT') { + if (content && content.kind === Kind.Metadata) { + setData(key, 'users', (users) => ({ ...users, [content.pubkey]: content})) + } + if (content && (content.kind === Kind.Text || content.kind === Kind.Repost)) { + setData(key, 'messages', (msgs) => [ ...msgs, content]); + } + if (content && content.kind === Kind.NoteStats) { + const stat = JSON.parse(content.content); + setData(key, 'postStats', (stats) => ({ ...stats, [stat.event_id]: stat })) + } + if (content && content.kind === Kind.Mentions) { + const mentionContent = content as NostrMentionContent; + const mention = JSON.parse(mentionContent.content); + + setData(key, 'mentions', + (mentions) => ({ ...mentions, [mention.id]: { ...mention } }) + ); + return; + } + } + }; + +// SOCKET HANDLERS ------------------------------ + + const onSocketClose = (closeEvent: CloseEvent) => { + const webSocket = closeEvent.target as WebSocket; + + webSocket.removeEventListener('message', onMessage); + webSocket.removeEventListener('close', onSocketClose); + }; + + const onMessage = (event: MessageEvent) => { + const message: NostrEvent | NostrEOSE = JSON.parse(event.data); + + const [type, subId, content] = message; + + if (subId === `sidebar_trending_${APP_ID}`) { + processNotes(type, 'trending', content); + return; + } + if (subId === `sidebar_zapped_${APP_ID}`) { + processNotes(type, 'mostzapped', content); + return; + } + }; + + return ( +
+
+
+
+ {intl.formatMessage({ + id: 'home.sidebar.caption.trending', + defaultMessage: 'Trending', + description: 'Caption for the home page sidebar showing a list of trending notes', + })} + + {intl.formatNumber(24, hourNarrow)} + +
+
+ + + {(note) => } + + +
+
+
+ {intl.formatMessage({ + id: 'home.sidebar.caption.mostzapped', + defaultMessage: 'Most Zapped', + description: 'Caption for the home page sidebar showing a list of most zapped notes', + })} + + {intl.formatNumber(4, hourNarrow)} + +
+
+ + { + (note) => + + {intl.formatMessage({ + id: 'home.sidebar.note.zaps', + defaultMessage: '{zaps} zaps, {sats} sats', + description: 'Zaps data for a small note on home sidebar', + }, + { + zaps: humanizeNumber(note.post.zaps, true), + sats: humanizeNumber(note.post.satszapped, true), + })} + + } + +
+ ); +} + +export default HomeSidebar; diff --git a/src/components/Layout/Layout.module.scss b/src/components/Layout/Layout.module.scss new file mode 100644 index 0000000..9c6727d --- /dev/null +++ b/src/components/Layout/Layout.module.scss @@ -0,0 +1,202 @@ +.container { + width: 1240px; + // height: 100vh; + margin: 0px auto; + + display: grid; + grid-template-columns: 176px 640px 300px; + grid-column-gap: 32px; +} + +.leftColumn { + + >div { + position: fixed; + width: 176px; + display: grid; + grid-template-rows: 128px 1fr 82px; + height: 100%; + } + + .leftHeader { + height: 128px; + display: grid; + align-items: center; + justify-content: right; + } + + .leftContent { + display: grid; + height: 100%; + justify-content: right; + } + + .leftFooter { + height: 82px; + display: flex; + justify-content: flex-end; + } +} + + +.centerColumn { + display: grid; + grid-template-rows: 128px 1fr; + position: relative; +} + +.centerHeader { + width: 640px; +} + +.centerContent { + width: 640px; + .headerFloater { + position: fixed; + opacity: 0; + pointer-events: none; + width: 640px; + z-index: var(--z-index-floater); + + &.animatedShow { + opacity: 1; + transition: opacity 0.5s ease; + pointer-events: all; + } + } +} + +.rightColumn { + display: grid; + width: 300px; + grid-template-rows: 128px 1fr; + grid-row-gap: 28px; +} + +.rightHeader { + height: 128px; + display: grid; + align-items: center; + justify-content: left; + position: fixed; + // background-color: var(--background-site); + z-index: var(--z-index-header); +} + +.rightContent { + margin-top: 128px; +} + +.modal { + position: fixed; + z-index: 1200; +} + +.preload { + width: 0px; + height: 0px; + position: absolute; + top:0; + left:0; +} + +@media only screen and (max-width: 1300px) { + .container { + width: 1032px; + grid-template-columns: 48px 640px 300px; + } + + .leftColumn { + >div { + position: fixed; + width: 48px; + } + } + + .rightColumn { + width: 300px; + } +} + +@media only screen and (max-width: 1087px) { + .container { + width: 720px; + grid-template-columns: 48px 640px; + } + + .rightColumn { + display: none; + } +} + +@media only screen and (max-width: 720px) { + .container { + width: 100%; + // height: 100vh; + grid-template-columns: 1fr; + grid-template-rows: 1fr 48px; + grid-template-areas: "content" "footer"; + } + + .centerColumn { + grid-area: content; + width: 100%; + } + + .centerHeader { + width: 100%; + } + + .centerContent { + width: 100%; + + .headerFloater { + width: 100%; + } + } + + .leftColumn { + position: fixed; + left: 0px; + bottom: 0px; + width: 100%; + grid-area: footer; + display: flex; + flex-direction: row; + align-items: center; + justify-content: flex-start; + z-index: 20; + background-color: var(--background-site); + + >div { + position: relative; + height: 48px; + } + + .leftHeader { + display: none; + } + + .leftContent { + display: flex; + width: 100%; + } + + .leftFooter { + display: none; + } + + >div { + position: relative; + width: 100%; + } + } + + .rightColumn { + display: none; + } + + // body { + // overflow-x: hidden; + // } +} diff --git a/src/components/Layout/Layout.tsx b/src/components/Layout/Layout.tsx new file mode 100644 index 0000000..a93b027 --- /dev/null +++ b/src/components/Layout/Layout.tsx @@ -0,0 +1,112 @@ +import { Component, createEffect, onCleanup, onMount } from 'solid-js'; + +import styles from './Layout.module.scss'; + +import { Outlet } from '@solidjs/router'; +import NavMenu from '../NavMenu/NavMenu'; +import ProfileWidget from '../ProfileWidget/ProfileWidget'; +import NewNote from '../NewNote/NewNote'; +import { useAccountContext } from '../../contexts/AccountContext'; +import zapSM from '../../assets/lottie/zap_sm.json'; +import zapMD from '../../assets/lottie/zap_md.json'; + + +const Layout: Component = () => { + + const account = useAccountContext(); + + let container: HTMLDivElement | undefined; + + createEffect(() => { + const newNote = document.getElementById('new_note_input'); + const newNoteTextArea = document.getElementById('new_note_text_area') as HTMLTextAreaElement; + + if (account?.showNewNoteForm) { + newNote?.classList.add(styles.animatedShow); + newNoteTextArea?.focus(); + } + else { + newNote?.classList.remove(styles.animatedShow); + newNoteTextArea.value = ''; + } + }); + + const onResize = () => { + container?.style.setProperty('height', `${window.innerHeight}px`); + }; + + onMount(() => { + window.addEventListener('resize', onResize); + }); + + onCleanup(() => { + window.removeEventListener('resize', onResize); + }); + + return ( + <> +
+
+
+
+
+
+
+ + + + +
+ +
+
+
+
+
+ +
+ +
+ +
+ +
+
+
+ + +
+
+
+ +
+ +
+ +
+
+
+ + +
+
+
+
+
+
+ +
+
+
+ + ) +} + +export default Layout; diff --git a/src/components/LinkPreview/LinkPreview.module.scss b/src/components/LinkPreview/LinkPreview.module.scss new file mode 100644 index 0000000..da1f7b7 --- /dev/null +++ b/src/components/LinkPreview/LinkPreview.module.scss @@ -0,0 +1,23 @@ +.linkPreview { + width: 100%; + text-decoration: none; + color: var(--text-tertiary-2); + font-size: 16px; + font-weight: 400; + line-height: 22px; + text-align: left; +} + +.previewInfo { + padding-inline: 14px; + padding-block: 9px; +} + +.previewImage { + overflow: hidden; + max-height: 250px; +} + +.previewTitle { + color: var(--text-primary); +} diff --git a/src/components/LinkPreview/LinkPreview.tsx b/src/components/LinkPreview/LinkPreview.tsx new file mode 100644 index 0000000..14c8202 --- /dev/null +++ b/src/components/LinkPreview/LinkPreview.tsx @@ -0,0 +1,36 @@ +import { A, useLocation, useNavigate } from '@solidjs/router'; +import { Component, Show } from 'solid-js'; + +import styles from './LinkPreview.module.scss'; + +const LinkPreview: Component<{ preview: any }> = (props) => { + + return ( + + +
+ +
+
+ +
+ +
{props.preview.url}
+
+ + +
{props.preview.title}
+
+ + +
{props.preview.description}
+
+
+
+ ); +} + +export default LinkPreview; diff --git a/src/components/Loader/Loader.module.scss b/src/components/Loader/Loader.module.scss new file mode 100644 index 0000000..e0a9d54 --- /dev/null +++ b/src/components/Loader/Loader.module.scss @@ -0,0 +1,51 @@ +.loader { + height: 32px; + width: 32px; +} +.loader span { + display: block; + position: absolute; + top: 0; left: 0; + bottom: 0; right: 0; + margin: auto; + height: 32px; + width: 32px; +} + +.loader span::before, +.loader span::after { + content: ""; + display: block; + position: absolute; + top: 0; left: 0; + bottom: 0; right: 0; + margin: auto; + height: 32px; + width: 32px; + border: 2px solid var(--brand-1); + border-radius: 50%; + opacity: 0; + -webkit-animation: loader-1 1.5s cubic-bezier(0.075, 0.820, 0.165, 1.000) infinite; + animation: loader-1 1.5s cubic-bezier(0.075, 0.820, 0.165, 1.000) infinite; +} +@-webkit-keyframes loader-1 { + 0% { -webkit-transform: translate3d(0, 0, 0) scale(0); opacity: 1; } + 100% { -webkit-transform: translate3d(0, 0, 0) scale(1.5); opacity: 0; } +} +@keyframes loader-1 { + 0% { transform: translate3d(0, 0, 0) scale(0); opacity: 1; } + 100% { transform: translate3d(0, 0, 0) scale(1.5); opacity: 0; } +} + +// .loader span::after { +// -webkit-animation: loader-2 1.5s cubic-bezier(0.075, 0.820, 0.165, 1.000) .25s infinite; +// animation: loader-2 1.5s cubic-bezier(0.075, 0.820, 0.165, 1.000) .25s infinite; +// } +// @-webkit-keyframes loader-2 { +// 0% { -webkit-transform: translate3d(0, 0, 0) scale(0); opacity: 1; } +// 100% { -webkit-transform: translate3d(0, 0, 0) scale(1); opacity: 0; } +// } +// @keyframes loader-2 { +// 0% { transform: translate3d(0, 0, 0) scale(0); opacity: 1; } +// 100% { transform: translate3d(0, 0, 0) scale(1); opacity: 0; } +// } diff --git a/src/components/Loader/Loader.tsx b/src/components/Loader/Loader.tsx new file mode 100644 index 0000000..7741ae9 --- /dev/null +++ b/src/components/Loader/Loader.tsx @@ -0,0 +1,12 @@ +import { Component } from 'solid-js'; + +import styles from './Loader.module.scss'; + +const Loader: Component = () => { + + return ( +
+ ); +} + +export default Loader; diff --git a/src/components/MissingPage/MissingPage.module.scss b/src/components/MissingPage/MissingPage.module.scss new file mode 100644 index 0000000..74d8360 --- /dev/null +++ b/src/components/MissingPage/MissingPage.module.scss @@ -0,0 +1,21 @@ +.fullHeader { + display: grid; + height: 120px; + align-items: center; + justify-content: left; + + >div { + font-weight: 300; + font-size: 32px; + line-height: 34px; + color: var(--brand-text); + text-transform: lowercase; + } +} + +.comingSoon { + font-weight: 300; + font-size: 18px; + line-height: 34px; + color: var(--text-secondary); +} diff --git a/src/components/MissingPage/MissingPage.tsx b/src/components/MissingPage/MissingPage.tsx new file mode 100644 index 0000000..5b488a5 --- /dev/null +++ b/src/components/MissingPage/MissingPage.tsx @@ -0,0 +1,51 @@ +import { useIntl } from '@cookbook/solid-intl'; +import { Component, JSXElement, Show } from 'solid-js'; +import { placeholders as t } from '../../translations'; +import Branding from '../Branding/Branding'; +import Search from '../Search/Search'; +import Wormhole from '../Wormhole/Wormhole'; +import styles from './MissingPage.module.scss'; + + +const MissingPage: Component<{ title: string, children?: JSXElement }> = (props) => { + + const intl = useIntl(); + + return ( + <> + + + + + + + + +
+
+ {intl.formatMessage( + t.pageWIPTitle, + { title: props.title }, + )} +
+
+ + + {intl.formatMessage(t.comingSoon)} + + } + > +
+ {props.children} +
+
+ + ) +} + +export default MissingPage; diff --git a/src/components/Modal/Modal.module.scss b/src/components/Modal/Modal.module.scss new file mode 100644 index 0000000..f5e6042 --- /dev/null +++ b/src/components/Modal/Modal.module.scss @@ -0,0 +1,8 @@ +.modal { + background-color: var(--background-modal); + width: 100vw; + height:100vh; + display: flex; + align-items: center; + justify-content: center; +} diff --git a/src/components/Modal/Modal.tsx b/src/components/Modal/Modal.tsx new file mode 100644 index 0000000..7374fbd --- /dev/null +++ b/src/components/Modal/Modal.tsx @@ -0,0 +1,19 @@ +import { Component, JSXElement, Show } from 'solid-js'; +import { Portal } from 'solid-js/web'; + +import styles from './Modal.module.scss'; + +const Modal: Component<{ children: JSXElement, open?: boolean}> = (props) => { + + return ( + + +
+ {props.children} +
+
+
+ ); +} + +export default Modal; diff --git a/src/components/NavLink/NavLink.module.scss b/src/components/NavLink/NavLink.module.scss new file mode 100644 index 0000000..a2185a0 --- /dev/null +++ b/src/components/NavLink/NavLink.module.scss @@ -0,0 +1,148 @@ +@mixin iconNav { + width: 32px; + height: 32px; + background-color: var(--text-secondary); +} + +.navLink { + position: relative; + text-align: right; + margin: 0px 0px 28px 0px; + padding: 0px; + background: none; + border: none; + border-radius: 0px; + + div { + display: inline; + } + + p { + display: inline; + text-transform: uppercase; + font-size: 18px; + line-height: 20px; + font-weight: 800; + } +} + +.homeIcon { + @include iconNav; + -webkit-mask: url(../../assets/icons/home.svg) no-repeat center; + mask: url(../../assets/icons/home.svg) no-repeat center; +} + +.exploreIcon { + @include iconNav; + -webkit-mask: url(../../assets/icons/explore.svg) no-repeat center; + mask: url(../../assets/icons/explore.svg) no-repeat center; +} +.messagesIcon { + @include iconNav; + -webkit-mask: url(../../assets/icons/messages.svg) no-repeat center; + mask: url(../../assets/icons/messages.svg) no-repeat center; +} +.notificationsIcon { + @include iconNav; + -webkit-mask: url(../../assets/icons/notifications.svg) no-repeat center; + mask: url(../../assets/icons/notifications.svg) no-repeat center; +} +.downloadIcon { + @include iconNav; + -webkit-mask: url(../../assets/icons/download.svg) no-repeat center; + mask: url(../../assets/icons/download.svg) no-repeat center; +} +.settingsIcon { + @include iconNav; + -webkit-mask: url(../../assets/icons/settings.svg) no-repeat center; + mask: url(../../assets/icons/settings.svg) no-repeat center; +} +.helpIcon { + @include iconNav; + -webkit-mask: url(../../assets/icons/help.svg) no-repeat center; + mask: url(../../assets/icons/help.svg) no-repeat center; +} + +.active { + p { + color: var(--active-link); + text-decoration: underline; + } + div { + background-color: var(--active-link); + } + + .homeIcon { + background-color: var(--text-primary); + } + .exploreIcon { + background-color: var(--text-primary); + } + .messagesIcon { + background-color: var(--text-primary); + } + .notificationsIcon { + background-color: var(--text-primary); + } + .downloadIcon { + background-color: var(--text-primary); + } + .settingsIcon { + background-color: var(--text-primary); + } + .helpIcon { + background-color: var(--text-primary); + } +} + +.inactive { + p { + color: var(--inactive-link); + } + div { + background-color: var(--inactive-link); + } +} + +@media only screen and (max-width: 1300px) { + .navLink { + margin-bottom: 22px; + + width: fit-content; + div { + display: inline-block; + } + + p { + display: none; + } + } +} + +.bubble { + position: absolute; + text-align: center; + padding-top: 2px; + padding-inline: 4px; + top: 0; + right: -18px; + min-width: 18px; + min-height: 18px; + border-radius: 8px; + font-weight: 500; + font-size: 12px; + line-height: 12px; + + background: var(--brand-gradient); + border: 1px solid var(--background-site); + + color: white; + text-shadow: 0.5px 0.5px 0px black; + + &.doubleSize { + right: -24px; + } + &.tripleSize { + right: -30px; + } +} diff --git a/src/components/NavLink/NavLink.tsx b/src/components/NavLink/NavLink.tsx new file mode 100644 index 0000000..0bc005d --- /dev/null +++ b/src/components/NavLink/NavLink.tsx @@ -0,0 +1,51 @@ +import { A, useLocation, useNavigate } from '@solidjs/router'; +import { Component, Show } from 'solid-js'; + +import styles from './NavLink.module.scss'; + +const NavLink: Component<{ to: string, label: string, icon: string, bubble?: () => number}> = (props) => { + + const navigate = useNavigate(); + const location = useLocation(); + + const shouldScroll = () => props.to === location.pathname; + + const onClick = (e: Event) => { + if (shouldScroll()) { + e.preventDefault(); + + window.scrollTo({ top: 0, left: 0, behavior: 'smooth' }); + return; + } + + navigate('/home'); + } + + const bubbleClass = () => { + if (!props.bubble || props.bubble() < 10) { + return ''; + } + + if (props.bubble() < 100) { + return styles.doubleSize; + } + + return styles.tripleSize; + } + + return ( + + ) +} + +export default NavLink; diff --git a/src/components/NavMenu/NavMenu.module.scss b/src/components/NavMenu/NavMenu.module.scss new file mode 100644 index 0000000..c1dd630 --- /dev/null +++ b/src/components/NavMenu/NavMenu.module.scss @@ -0,0 +1,43 @@ +.sideNav { + display: flex; + flex-direction: column; + + > div { + margin-bottom: 28px; + } +} + +.callToAction { + display: grid; + justify-content: right; + margin-top: 15px; +} + +@media only screen and (max-width: 720px) { + .navMenu { + width: 100%; + } + + .sideNav { + display: flex; + flex-direction: row; + justify-content: space-around; + align-items: flex-end; + width: 100%; + height: 48px; + + > button { + margin-bottom: 0px; + } + + button:nth-child(n+5) { + display: none; + } + } + + .callToAction { + position: fixed; + bottom: 56px; + right: 24px; + } +} diff --git a/src/components/NavMenu/NavMenu.tsx b/src/components/NavMenu/NavMenu.tsx new file mode 100644 index 0000000..66d2043 --- /dev/null +++ b/src/components/NavMenu/NavMenu.tsx @@ -0,0 +1,79 @@ +import { useIntl } from '@cookbook/solid-intl'; +import { useLocation } from '@solidjs/router'; +import { Component, For, Show } from 'solid-js'; +import { useAccountContext } from '../../contexts/AccountContext'; +import { useMessagesContext } from '../../contexts/MessagesContext'; +import { useNotificationsContext } from '../../contexts/NotificationsContext'; +import { navBar as t } from '../../translations'; +import NavLink from '../NavLink/NavLink'; +import FloatingNewPostButton from '../FloatingNewPostButton/FloatingNewPostButton'; + +import styles from './NavMenu.module.scss'; + +const NavMenu: Component = () => { + const account = useAccountContext(); + const notifications = useNotificationsContext(); + const messages = useMessagesContext(); + const intl = useIntl(); + const loc = useLocation(); + + const links = [ + { + to: '/home', + label: intl.formatMessage(t.home), + icon: 'homeIcon', + }, + { + to: '/explore', + label: intl.formatMessage(t.explore), + icon: 'exploreIcon', + }, + { + to: '/messages', + label: intl.formatMessage(t.messages), + icon: 'messagesIcon', + bubble: () => messages?.messageCount || 0, + }, + { + to: '/notifications', + label: intl.formatMessage(t.notifications), + icon: 'notificationsIcon', + bubble: () => notifications?.notificationCount || 0, + }, + { + to: '/downloads', + label: intl.formatMessage(t.downloads), + icon: 'downloadIcon', + }, + { + to: '/settings', + label: intl.formatMessage(t.settings), + icon: 'settingsIcon', + }, + { + to: '/help', + label: intl.formatMessage(t.help), + icon: 'helpIcon', + }, + ]; + + return ( +
+ + +
+ +
+
+
+ ) +} + +export default NavMenu; diff --git a/src/components/NewNote/EditBox/EditBox.module.scss b/src/components/NewNote/EditBox/EditBox.module.scss new file mode 100644 index 0000000..9f48742 --- /dev/null +++ b/src/components/NewNote/EditBox/EditBox.module.scss @@ -0,0 +1,320 @@ +.noteEditBox { + position: relative; +} + +.newNoteHolder { + min-height: 100px; + max-height: 100vh; + background-color: var(--background-site); + padding-top: 10px; +} + +.holderBottomBorder { + width: 100%; + height: 4px; + display: flex; + justify-content: space-between; + + .rightCorner { + display: inline-block; + width: 4px; + height: 4px; + + background-color: var(--background-site); + -webkit-mask: url(../../assets/icons/corner_right.svg) no-repeat center; + mask: url(../../assets/icons/corner_right.svg) no-repeat center; + } + .leftCorner { + display: inline-block; + width: 4px; + height: 4px; + + background-color: var(--background-site); + -webkit-mask: url(../../assets/icons/corner_left.svg) no-repeat center; + mask: url(../../assets/icons/corner_left.svg) no-repeat center; + } +} + +.newNoteBorder { + width: 100%; + height: 100%; + padding: 1px; + background: var(--brand-gradient); + border-radius: 6px; + display: block; + position: relative; +} + +.newNote { + width: 100%; + height: 100%; + min-height: 122px; + font-size: 18px; + line-height: 20px; + margin: 0px; + border-radius: 6px; + border: none; + color: var(--text-tertiary); + background-color: var(--background-site); + padding-bottom: 11px; + display: grid; + grid-template-columns: 92px 1fr; + + .leftSide { + padding: 20px; + } + +} + +.controls { + position: absolute; + bottom: 4px; + right: 15px; + display: flex; + align-items: center; + justify-content: flex-end; + width: calc(100% - 16px); + // border: solid 1px red; + padding-top: 8px; + background-color: var(--background-site); + >button { + width: 80px; + height: 28px; + min-width: 80px; + margin: 0px 0px 11px 8px; + } + .editorOptions { + width: 100%; + // padding-left: 80px; + + .attachIcon { + width: 26px; + height: 21px; + display: inline-block; + margin-right: 9px; + background-color: var(--text-tertiary); + cursor: pointer; + + &:hover { + background-color: var(--text-secondary); + } + } + + } + +} + +.primaryButton { + border: none; + border-radius: 6px; + margin: 0px 8px; + padding: 0px; + font-size: 14px; + line-height: 20px; + font-weight: 700; + background: var(--brand-gradient-vertical); + color: white; + >span { + opacity: 0.75; + } +} + + +.secondaryButton { + border: none; + border-radius: 6px; + padding: 1px; + font-size: 14px; + line-height: 20px; + font-weight: 700; + background: var(--brand-gradient-vertical); + color: var(--text-tertiary-2); + >div { + width: 100%; + height: 100%; + vertical-align: middle; + border-radius: 6px; + background-color: var(--background-card); + display: flex; + flex-direction: column; + align-items: center; + justify-content: center; + } +} + +.searchSuggestions { + width: 300px; + background-color: var(--background-site); + border: 1px solid var(--text-tertiary-2); + // box-shadow: 0px 4px 12px rgba(0, 0, 0, 0.8); + border-radius: 4px; + + position: absolute; + top: 0px; + left: 0px; + z-index: var(--z-index-floater); +} + +.highlight { + color: var(--accent-1); +} + +.userReference { + color: var(--accent-1); + &:hover { + text-decoration: underline; + } +} + +textarea { + width: calc(100% - 18px); + max-height: calc(100vh / 3); + margin: 18px 18px 36px 0px; + padding: 0px; + box-sizing: padding-box; + border: none; + border-radius: 0px; + background-color: unset; + font-size: 16px; + line-height: 20px; + font-weight: 400; + color: var(--text-primary); + resize: none; + scrollbar-width: none; /* Firefox */ + + &:focus { + border: none; + outline: none; + } +} + +textarea::-webkit-scrollbar{ + display: none; /* Safari and Chrome */ + } + +.error { + color: var(--brand-1); +} + +.editorWrap { + display: flex; + flex-direction: column; + flex: 1; + cursor: text; + // max-height: calc(100vh - 7px); +} + +.previewCaption { + color: var(--subtile-devider); + font-weight: 400; + font-size: 10px; + line-height: 16px; + text-transform: uppercase; +} + +.editorScroll { + min-height: 60px; + max-height: calc(60vh - 70px); + overflow-y: scroll; + margin-bottom: 48px; + + scrollbar-width: none; /* Firefox */ + +} + +.editorScroll::-webkit-scrollbar { + display: none; /* Safari and Chrome */ +} + +.editor { + max-width: calc(100% - 14px); + min-height: 60px; + outline: 0px solid transparent; + word-wrap: break-word; + word-break: break-all; + font-size: 16px; + line-height: 20px; + font-weight: 400; + color: var(--text-primary); + margin: 0px 14px 32px 0px; + padding: 0px; + background-color: var(--background-card); + border-radius: 4px; + padding: 12px; +} + + +.emojiSuggestions { + position: absolute; + display: grid; + grid-template-columns: 50px 50px 50px 50px 50px 50px; + width: 322px; + max-height: 200px; + overflow-y: scroll; + padding: 4px; + background-color: var(--background-site); + border: 1px solid var(--text-tertiary-2); + // box-shadow: 0px 4px 12px rgba(0, 0, 0, 0.8); + border-radius: 8px; + + top: 0px; + left: 0px; + z-index: var(--z-index-floater); + + .emojiOption { + margin-bottom: 5px; + padding: 2px; + background: none; + font-size: 16px; + line-height: 20px; + font-weight: 400; + border: none; + display: flex; + justify-content: center; + align-items: center; + + &:hover, &.highlight { + background-color: var(--text-tertiary-2); + } + + &:focus { + outline: none; + border: none; + } + } +} + +.uploadLoader { + position: absolute; + pointer-events: none; + top: 0; + left: 0; + width: 100%; + height: 100%; + background-color: var(--background-site); + opacity: 0.8; + display: flex; + flex-direction: column; + align-items: center; + justify-content: center; + color: var(--text-secondary); + > div { + position: relative; + min-height: 48px; + } +} + +.dropOverlay { + position: absolute; + top: 0; + left: 0; + width: 100%; + height: 100%; + background-color: var(--background-site); + opacity: 0.8; + display: flex; + flex-direction: column; + align-items: center; + justify-content: center; + color: var(--text-secondary); +} diff --git a/src/components/NewNote/EditBox/EditBox.tsx b/src/components/NewNote/EditBox/EditBox.tsx new file mode 100644 index 0000000..ac20bdb --- /dev/null +++ b/src/components/NewNote/EditBox/EditBox.tsx @@ -0,0 +1,1163 @@ +import { useIntl } from "@cookbook/solid-intl"; +import { Router, useLocation } from "@solidjs/router"; +import { nip19 } from "nostr-tools"; +import { Component, createEffect, createSignal, For, onCleanup, onMount, Show } from "solid-js"; +import { createStore } from "solid-js/store"; +import { noteRegex, profileRegex, Kind, editMentionRegex, emojiSearchLimit } from "../../../constants"; +import { useAccountContext } from "../../../contexts/AccountContext"; +import { useSearchContext } from "../../../contexts/SearchContext"; +import { TranslatorProvider } from "../../../contexts/TranslatorContext"; +import { getEvents } from "../../../lib/feed"; +import { parseNote1, sanitize, sendNote, replaceLinkPreviews } from "../../../lib/notes"; +import { getUserProfiles } from "../../../lib/profile"; +import { subscribeTo } from "../../../sockets"; +import { subscribeTo as uploadSub } from "../../../uploadSocket"; +import { convertToNotes, referencesToTags } from "../../../stores/note"; +import { convertToUser, nip05Verification, truncateNpub, userName } from "../../../stores/profile"; +import { EmojiOption, FeedPage, NostrMediaUploaded, NostrMentionContent, NostrNoteContent, NostrStatsContent, NostrUserContent, PrimalNote, PrimalUser } from "../../../types/primal"; +import { debounce, isVisibleInContainer, uuidv4 } from "../../../utils"; +import Avatar from "../../Avatar/Avatar"; +import EmbeddedNote from "../../EmbeddedNote/EmbeddedNote"; +import MentionedUserLink from "../../Note/MentionedUserLink/MentionedUserLink"; +import SearchOption from "../../Search/SearchOption"; +import { useToastContext } from "../../Toaster/Toaster"; +import styles from './EditBox.module.scss'; +import emojiSearch from '@jukben/emoji-search'; +import { getCaretCoordinates } from "../../../lib/textArea"; +import { uploadMedia } from "../../../lib/media"; +import { APP_ID } from "../../../App"; +import Loader from "../../Loader/Loader"; +import { + toast as tToast, + feedback as tFeedback, + note as tNote, + search as tSearch, + actions as tActions, +} from "../../../translations"; + +type AutoSizedTextArea = HTMLTextAreaElement & { _baseScrollHeight: number }; + + +const EditBox: Component<{ replyToNote?: PrimalNote, onClose?: () => void, idPrefix?: string } > = (props) => { + + const intl = useIntl(); + + const instanceId = uuidv4(); + + const search = useSearchContext(); + const account = useAccountContext(); + const toast = useToastContext(); + + let textArea: HTMLTextAreaElement | undefined; + let textPreview: HTMLDivElement | undefined; + let mentionOptions: HTMLDivElement | undefined; + let emojiOptions: HTMLDivElement | undefined; + let editWrap: HTMLDivElement | undefined; + let fileUpload: HTMLInputElement | undefined; + + let mentionCursorPosition = { top: 0, left: 0, height: 0 }; + let emojiCursorPosition = { top: 0, left: 0, height: 0 }; + + const [isMentioning, setMentioning] = createSignal(false); + const [preQuery, setPreQuery] = createSignal(''); + const [query, setQuery] = createSignal(''); + + const [message, setMessage] = createSignal(''); + const [parsedMessage, setParsedMessage] = createSignal(''); + + const [isEmojiInput, setEmojiInput] = createSignal(false); + const [emojiQuery, setEmojiQuery] = createSignal(''); + const [emojiResults, setEmojiResults] = createStore([]); + + const [userRefs, setUserRefs] = createStore>({}); + const [noteRefs, setNoteRefs] = createStore>({}); + + const [highlightedUser, setHighlightedUser] = createSignal(0); + const [highlightedEmoji, setHighlightedEmoji] = createSignal(0); + const [referencedNotes, setReferencedNotes] = createStore>(); + + const location = useLocation(); + + let currentPath = location.pathname; + + const getScrollHeight = (elm: AutoSizedTextArea) => { + var savedValue = elm.value + elm.value = '' + elm._baseScrollHeight = elm.scrollHeight + elm.value = savedValue + } + + const onExpandableTextareaInput: (event: InputEvent) => void = (event) => { + const maxHeight = document.documentElement.clientHeight || window.innerHeight || 0; + + const elm = textArea as AutoSizedTextArea; + const preview = textPreview; + + if(elm.nodeName !== 'TEXTAREA' || elm.id !== `${prefix()}new_note_text_area` || !preview) { + return; + } + + const minRows = parseInt(elm.getAttribute('data-min-rows') || '0'); + + !elm._baseScrollHeight && getScrollHeight(elm); + + + if (elm.scrollHeight >= (maxHeight / 3)) { + return; + } + + elm.rows = minRows; + const rows = Math.ceil((elm.scrollHeight - elm._baseScrollHeight) / 20); + elm.rows = minRows + rows; + + const rect = elm.getBoundingClientRect(); + + preview.style.maxHeight = `${maxHeight - rect.height - 120}px`; + } + + createEffect(() => { + if (emojiQuery().length > emojiSearchLimit) { + setEmojiResults(() => emojiSearch(emojiQuery())); + } + }); + + + createEffect(() => { + if (isEmojiInput() && emojiQuery().length > emojiSearchLimit) { + emojiPositionOptions(); + } + }); + + const onKeyDown = (e: KeyboardEvent) => { + if (!textArea) { + return false; + } + + if (isUploading()) { + return; + } + + const mentionSeparators = ['Enter', 'Space', 'Comma']; + + if (e.code === 'Enter' && e.metaKey) { + e.preventDefault(); + postNote(); + return false; + } + + if (!isMentioning() && !isEmojiInput() && e.key === ':') { + emojiCursorPosition = getCaretCoordinates(textArea, textArea.selectionStart); + setEmojiInput(true); + return false; + } + + if (isEmojiInput()) { + + if (e.code === 'ArrowDown') { + e.preventDefault(); + setHighlightedEmoji(i => { + if (emojiResults.length === 0) { + return 0; + } + + return i < emojiResults.length - 7 ? i + 6 : 0; + }); + + const emojiHolder = document.getElementById(`${instanceId}-${highlightedEmoji()}`); + + if (emojiHolder && emojiOptions && !isVisibleInContainer(emojiHolder, emojiOptions)) { + emojiHolder.scrollIntoView({ block: 'end', behavior: 'smooth' }); + } + + return false; + } + + if (e.code === 'ArrowUp') { + e.preventDefault(); + setHighlightedEmoji(i => { + if (emojiResults.length === 0) { + return 0; + } + + return i >= 6 ? i - 6 : emojiResults.length - 1; + }); + + const emojiHolder = document.getElementById(`${instanceId}-${highlightedEmoji()}`); + + if (emojiHolder && emojiOptions && !isVisibleInContainer(emojiHolder, emojiOptions)) { + emojiHolder.scrollIntoView({ block: 'start', behavior: 'smooth' }); + } + + return false; + } + + if (e.code === 'ArrowRight') { + e.preventDefault(); + setHighlightedEmoji(i => { + if (emojiResults.length === 0) { + return 0; + } + + return i < emojiResults.length - 1 ? i + 1 : 0; + }); + + const emojiHolder = document.getElementById(`${instanceId}-${highlightedEmoji()}`); + + if (emojiHolder && emojiOptions && !isVisibleInContainer(emojiHolder, emojiOptions)) { + emojiHolder.scrollIntoView({ block: 'end', behavior: 'smooth' }); + } + + return false; + } + + if (e.code === 'ArrowLeft') { + e.preventDefault(); + setHighlightedEmoji(i => { + if (emojiResults.length === 0) { + return 0; + } + + return i > 0 ? i - 1 : emojiResults.length - 1; + }); + + const emojiHolder = document.getElementById(`${instanceId}-${highlightedEmoji()}`); + + if (emojiHolder && emojiOptions && !isVisibleInContainer(emojiHolder, emojiOptions)) { + emojiHolder.scrollIntoView({ block: 'start', behavior: 'smooth' }); + } + + return false; + } + + if (mentionSeparators.includes(e.code)) { + if (emojiQuery().trim().length === 0) { + setEmojiInput(false); + return false; + } + e.preventDefault(); + selectEmoji(emojiResults[highlightedEmoji()]); + setHighlightedEmoji(0); + return false; + } + + const cursor = textArea.selectionStart; + const lastEmojiTrigger = textArea.value.slice(0, cursor).lastIndexOf(':'); + + if (e.code === 'Backspace') { + setEmojiQuery(emojiQuery().slice(0, -1)); + + if (lastEmojiTrigger < 0 || cursor - lastEmojiTrigger <= 1) { + setEmojiInput(false); + return false; + } + } else { + setEmojiQuery(q => q + e.key); + return false; + } + + return false; + } + + if (!isMentioning() && e.key === '@') { + mentionCursorPosition = getCaretCoordinates(textArea, textArea.selectionStart); + setPreQuery(''); + setQuery(''); + setMentioning(true); + return false; + } + + if (!isMentioning() && e.code === 'Backspace' && textArea) { + let cursor = textArea.selectionStart; + const textSoFar = textArea.value.slice(0, cursor); + const lastWord = textSoFar.split(/[\s,;\n\r]/).pop(); + + if (lastWord?.startsWith('@`')) { + const index = textSoFar.lastIndexOf(lastWord); + + const newText = textSoFar.slice(0, index) + textArea.value.slice(cursor); + + setMessage(newText); + textArea.value = newText; + + textArea.selectionEnd = index; + } + } + + if (isMentioning()) { + + if (e.code === 'ArrowDown') { + e.preventDefault(); + setHighlightedUser(i => { + if (!search?.users || search.users.length === 0) { + return 0; + } + + return i < search.users.length - 1 ? i + 1 : 0; + }); + return false; + } + + if (e.code === 'ArrowUp') { + e.preventDefault(); + setHighlightedUser(i => { + if (!search?.users || search.users.length === 0) { + return 0; + } + + return i > 0 ? i - 1 : search.users.length - 1; + }); + return false; + } + + if (mentionSeparators.includes(e.code)) { + if (preQuery() === ' ') { + setMentioning(false); + return false; + } + e.preventDefault(); + search?.users && selectUser(search.users[highlightedUser()]) + setMentioning(false); + return false; + } + + const cursor = textArea.selectionStart; + const lastMentionTrigger = textArea.value.slice(0, cursor).lastIndexOf('@'); + + if (e.code === 'Backspace') { + setPreQuery(preQuery().slice(0, -1)); + + if (lastMentionTrigger < 0 || cursor - lastMentionTrigger <= 1) { + setMentioning(false); + return false; + } + } else { + setPreQuery(q => q + e.key); + return false + } + + return false; + } + + return true; + }; + + const [isDroppable, setIsDroppable] = createSignal(false); + + const onDrop = (e: DragEvent) => { + e.preventDefault(); + e.stopPropagation(); + + setIsDroppable(false); + + let draggedData = e.dataTransfer; + let file = draggedData?.files[0]; + + + file && isSupportedFileType(file) && uploadFile(file); + + }; + + const onDragOver = (e: DragEvent) => { + e.preventDefault(); + e.stopPropagation(); + + setIsDroppable(true); + } + + const onDragLeave = (e: DragEvent) => { + e.preventDefault(); + e.stopPropagation(); + + if (!editWrap) { + return; + } + + const rect = editWrap.getBoundingClientRect(); + + const isWider = e.clientX < rect.x || e.clientX > rect.x + rect.width; + const isTaller = e.clientY < rect.y || e.clientY > rect.y + rect.height; + + (isWider || isTaller) && setIsDroppable(false); + } + + const onPaste = (e:ClipboardEvent) => { + if (e.clipboardData?.files && e.clipboardData.files.length > 0) { + e.preventDefault(); + const file = e.clipboardData.files[0]; + file && isSupportedFileType(file) && uploadFile(file); + return false; + } + } + + onMount(() => { + // @ts-expect-error TODO: fix types here + editWrap?.addEventListener('input', onExpandableTextareaInput); + editWrap?.addEventListener('keydown', onKeyDown); + // editWrap?.addEventListener('drop', onDrop, false); + }); + + onCleanup(() => { + // @ts-expect-error TODO: fix types here + editWrap?.removeEventListener('input', onExpandableTextareaInput); + editWrap?.removeEventListener('keydown', onKeyDown); + // editWrap?.removeEventListener('drop', onDrop, false); + }); + + createEffect(() => { + editWrap?.removeEventListener('keyup', onEscape); + editWrap?.addEventListener('keyup', onEscape); + }); + + createEffect(() => { + if (location.pathname !== currentPath) { + closeEditor(); + } + }) + + createEffect(() => { + const preQ = preQuery(); + + debounce(() => { + setQuery(() => preQ) + }, 500); + }) + + const onEscape = (e: KeyboardEvent) => { + if (e.code === 'Escape') { + !isMentioning() && !isEmojiInput() ? + closeEditor() : + closeEmojiAndMentions(); + } + }; + + const closeEditor = () => { + setUserRefs({}); + setMessage(''); + setParsedMessage(''); + setQuery(''); + setMentioning(false); + setEmojiInput(false); + setEmojiQuery('') + setEmojiResults(() => []); + props.onClose && props.onClose(); + }; + + const closeEmojiAndMentions = () => { + setMentioning(false); + setEmojiInput(false); + setEmojiQuery('') + setEmojiResults(() => []); + }; + + const postNote = async () => { + if (!account || !account.hasPublicKey() || isUploading() || isInputting()) { + return; + } + + if (Object.keys(account.relaySettings).length === 0) { + toast?.sendWarning( + intl.formatMessage(tToast.noRelays), + ); + return; + } + + if (account.relays.length === 0) { + toast?.sendWarning( + intl.formatMessage(tToast.noRelaysConnected), + ); + return; + } + + const value = message(); + + if (value.trim() === '') { + return; + } + + const messageToSend = value.replace(editMentionRegex, (url) => { + + const [_, name] = url.split('\`'); + const user = userRefs[name]; + + // @ts-ignore + return ` nostr:${user.npub}`; + }) + + if (account) { + const tags = referencesToTags(messageToSend); + + if (props.replyToNote) { + tags.push(['e', props.replyToNote.post.id, '', 'reply']); + tags.push(['p', props.replyToNote.post.pubkey]); + } + + const success = await sendNote(messageToSend, account.relays, tags); + + if (success) { + toast?.sendSuccess('Message posted successfully'); + } + else { + toast?.sendWarning('Failed to send message'); + } + } + + closeEditor(); + }; + + const mentionPositionOptions = () => { + if (!textArea || !mentionOptions || !editWrap) { + return; + } + + const taRect = textArea.getBoundingClientRect(); + const wRect = editWrap.getBoundingClientRect(); + + let newTop = taRect.top - wRect.top + mentionCursorPosition.top + 22; + let newLeft = mentionCursorPosition.left + 16; + + if (newTop > document.documentElement.clientHeight - 200) { + newTop = taRect.top - 400; + } + + mentionOptions.style.top = `${newTop}px`; + mentionOptions.style.left = `${newLeft}px`; + }; + + const emojiPositionOptions = () => { + if (!textArea || !emojiOptions || !editWrap) { + return; + } + + const taRect = textArea.getBoundingClientRect(); + const wRect = editWrap.getBoundingClientRect(); + + let newTop = taRect.top - wRect.top + emojiCursorPosition.top + 22; + let newLeft = emojiCursorPosition.left; + + if (newTop > document.documentElement.clientHeight - 200) { + newTop = taRect.top - 400; + } + + emojiOptions.style.top = `${newTop}px`; + emojiOptions.style.left = `${newLeft}px`; + }; + + const highlightHashtags = (text: string) => { + const regex = /(?:\s|^)#[^\s!@#$%^&*(),.?":{}|<>]+/ig; + + return text.replace(regex, (token) => { + const [space, term] = token.split('#'); + const embeded = ( + + {space} + + #{term} + + + ); + + // @ts-ignore + return embeded.outerHTML; + }); + } + + const parseUserMentions = (text: string) => { + return text.replace(editMentionRegex, (url) => { + const [_, name] = url.split('\`'); + const user = Object.values(userRefs).find(ref => userName(ref) === name); + + const link = user ? + MentionedUserLink({ user, openInNewTab: true}) : + @{name}; + + // @ts-ignore + return link.outerHTML || ` @${name}`; + }); + }; + + + const subUserRef = (userId: string) => { + + const parsed = parsedMessage().replace(profileRegex, (url) => { + const [_, id] = url.split(':'); + + if (!id) { + return url; + } + + try { + // const profileId = nip19.decode(id).data as string | nip19.ProfilePointer; + + // const hex = typeof profileId === 'string' ? profileId : profileId.pubkey; + // const npub = hexToNpub(hex); + + const user = userRefs[userId]; + + const link = user ? + @{userName(user)} : + @{truncateNpub(id)}; + + // @ts-ignore + return link.outerHTML || url; + } catch (e) { + return `${url}`; + } + }); + + setParsedMessage(parsed); + + }; + + const parseNpubLinks = (text: string) => { + let refs = []; + let match; + + while((match = profileRegex.exec(text)) !== null) { + refs.push(match[1]); + } + + refs.forEach(id => { + if (userRefs[id]) { + setTimeout(() => { + subUserRef(id); + }, 0); + return; + } + + const eventId = nip19.decode(id).data as string | nip19.ProfilePointer; + const hex = typeof eventId === 'string' ? eventId : eventId.pubkey; + + // setReferencedNotes(`nn_${id}`, { messages: [], users: {}, postStats: {}, mentions: {} }) + + const unsub = subscribeTo(`nu_${id}`, (type, subId, content) =>{ + if (type === 'EOSE') { + // // const newNote = convertToNotes(referencedNotes[subId])[0]; + + // // setNoteRefs((refs) => ({ + // // ...refs, + // // [newNote.post.noteId]: newNote + // // })); + + subUserRef(hex); + + unsub(); + return; + } + + if (type === 'EVENT') { + if (!content) { + return; + } + + if (content.kind === Kind.Metadata) { + const user = content as NostrUserContent; + + const u = convertToUser(user) + + setUserRefs(() => ({ [u.pubkey]: u })); + + // setReferencedNotes(subId, 'users', (usrs) => ({ ...usrs, [user.pubkey]: { ...user } })); + return; + } + } + }); + + + getUserProfiles([hex], `nu_${id}`); + + }); + + } + + const parseNoteLinks = (text: string) => { + let refs = []; + let match; + + while((match = noteRegex.exec(text)) !== null) { + refs.push(match[1]); + } + + refs.forEach(id => { + if (noteRefs[id]) { + setTimeout(() => { + subNoteRef(id); + }, 0); + return; + } + + const eventId = nip19.decode(id).data as string | nip19.EventPointer; + const hex = typeof eventId === 'string' ? eventId : eventId.id; + + setReferencedNotes(`nn_${id}`, { messages: [], users: {}, postStats: {}, mentions: {} }) + + const unsub = subscribeTo(`nn_${id}`, (type, subId, content) =>{ + if (type === 'EOSE') { + const newNote = convertToNotes(referencedNotes[subId])[0]; + + setNoteRefs((refs) => ({ + ...refs, + [newNote.post.noteId]: newNote + })); + + subNoteRef(newNote.post.noteId); + + unsub(); + return; + } + + if (type === 'EVENT') { + if (!content) { + return; + } + + if (content.kind === Kind.Metadata) { + const user = content as NostrUserContent; + + setReferencedNotes(subId, 'users', (usrs) => ({ ...usrs, [user.pubkey]: { ...user } })); + return; + } + + if ([Kind.Text, Kind.Repost].includes(content.kind)) { + const message = content as NostrNoteContent; + + setReferencedNotes(subId, 'messages', + (msgs) => [ ...msgs, { ...message }] + ); + + return; + } + + if (content.kind === Kind.NoteStats) { + const statistic = content as NostrStatsContent; + const stat = JSON.parse(statistic.content); + + setReferencedNotes(subId, 'postStats', + (stats) => ({ ...stats, [stat.event_id]: { ...stat } }) + ); + return; + } + + if (content.kind === Kind.Mentions) { + const mentionContent = content as NostrMentionContent; + const mention = JSON.parse(mentionContent.content); + + setReferencedNotes(subId, 'mentions', + (mentions) => ({ ...mentions, [mention.id]: { ...mention } }) + ); + return; + } + } + }); + + + getEvents(account?.publicKey, [hex], `nn_${id}`, true); + + }); + + }; + + const subNoteRef = (noteId: string) => { + + const parsed = parsedMessage().replace(noteRegex, (url) => { + const [_, id] = url.split(':'); + + if (!id || id !== noteId) { + return url; + } + try { + const note = noteRefs[id] + + const link = note ? +
+ + + + + +
: + {url}; + + // @ts-ignore + return link.outerHTML || url; + } catch (e) { + console.log('ERROR: ', e); + return `${url}`; + } + + }); + + setParsedMessage(parsed); + + }; + + + const parseForReferece = (value: string) => { + const content = replaceLinkPreviews(parseUserMentions(highlightHashtags(parseNote1(value)))); + + parseNpubLinks(content); + parseNoteLinks(content); + + return content; + }; + + const [isInputting, setIsInputting] = createSignal(false); + + const onInput = (e: InputEvent) => { + if (isUploading()) { + e.preventDefault(); + return false; + } + setIsInputting(true); + + // debounce(() => { + setIsInputting(false); + textArea && setMessage(textArea.value) + // }, 500) + }; + + let delayForMedia = 0; + + createEffect(() => { + if (delayForMedia) { + window.clearTimeout(delayForMedia); + } + const msg = sanitize(message()); + + delayForMedia = setTimeout(() => { + const p = parseForReferece(msg); + setParsedMessage(p); + }, 500); + + + }) + + createEffect(() => { + if (query().length === 0) { + search?.actions.getRecomendedUsers(); + return; + } + + search?.actions.findUsers(query()); + }); + + createEffect(() => { + if (isMentioning()) { + + mentionPositionOptions(); + + if (search?.users && search.users.length > 0) { + setHighlightedUser(0); + } + } + }); + + createEffect(() => { + if (isEmojiInput()) { + emojiPositionOptions(); + + if (emojiResults.length > 0) { + setHighlightedEmoji(0); + } + } + }); + + const selectEmoji = (emoji: EmojiOption) => { + if (!textArea) { + return; + } + + const msg = message(); + + // Get cursor position to determine insertion point + let cursor = textArea.selectionStart; + + // Get index of the token and insert emoji character + const index = msg.slice(0, cursor).lastIndexOf(':'); + const value = msg.slice(0, index) + emoji.char + msg.slice(cursor); + + // Reset query, update message and text area value + setMessage(value); + textArea.value = message(); + + // Calculate new cursor position + textArea.selectionEnd = index + 1; + textArea.focus(); + + setEmojiInput(false); + setEmojiQuery(''); + setEmojiResults(() => []); + + // Dispatch input event to recalculate UI position + // const e = new Event('input', { bubbles: true, cancelable: true}); + // textArea.dispatchEvent(e); + }; + + const selectUser = (user: PrimalUser | undefined) => { + if (!textArea || !user) { + return; + } + + setMentioning(false); + + const name = userName(user); + + setUserRefs((refs) => ({ + ...refs, + [name]: user, + })); + + const msg = message(); + + // Get cursor position to determine insertion point + let cursor = textArea.selectionStart; + + // Get index of the token and inster user's handle + const index = msg.slice(0, cursor).lastIndexOf('@'); + const value = msg.slice(0, index) + `@\`${name}\`` + msg.slice(cursor); + + // Reset query, update message and text area value + setQuery(''); + setMessage(value); + textArea.value = message(); + + textArea.focus(); + + // Calculate new cursor position + cursor = value.slice(0, cursor).lastIndexOf('@') + name.length + 3; + textArea.selectionEnd = cursor; + + + // Dispatch input event to recalculate UI position + const e = new Event('input', { bubbles: true, cancelable: true}); + textArea.dispatchEvent(e); + }; + + const focusInput = () => { + textArea && textArea.focus(); + }; + + const prefix = () => props.idPrefix ?? ''; + + const insertAtCursor = (text: string) => { + if (!textArea) { + return; + } + + const msg = message(); + + const cursor = textArea.selectionStart; + + const value = msg.slice(0, cursor) + `${text}` + msg.slice(cursor); + + setMessage(() => value); + textArea.value = value; + + textArea.focus(); + }; + + const [isUploading, setIsUploading] = createSignal(false); + + const isSupportedFileType = (file: File) => { + if (!file.type.startsWith('image/') && !file.type.startsWith('video/')) { + toast?.sendWarning(intl.formatMessage(tToast.fileTypeUpsupported)); + return false; + } + + return true; + + } + + const onUpload = () => { + if (!fileUpload) { + return; + } + + const file = fileUpload.files ? fileUpload.files[0] : null; + + // @ts-ignore fileUpload.value assignment + file && isSupportedFileType(file) && uploadFile(file, () => fileUpload.value = null); + + } + + const uploadFile = (file: File, callback?: () => void) => { + setIsUploading(true); + + const reader = new FileReader(); + + reader.onload = (e) => { + if (!e.target?.result) { + return; + } + + const subid = `upload_${APP_ID}`; + + const data = e.target?.result as string; + + const unsub = uploadSub(subid, (type, subId, content) => { + + if (type === 'EVENT') { + if (!content) { + return; + } + + if (content.kind === Kind.Uploaded) { + const uploaded = content as NostrMediaUploaded; + + insertAtCursor(uploaded.content); + return; + } + } + + if (type === 'NOTICE') { + setIsUploading(false); + unsub(); + return; + } + + if (type === 'EOSE') { + setIsUploading(false); + unsub(); + return; + } + }); + + uploadMedia(account?.publicKey, subid, data); + } + + reader.readAsDataURL(file); + + callback && callback(); + } + + return ( +
+ +
+ {intl.formatMessage(tFeedback.dropzone)} +
+
+ + +
+
+ +
+
{intl.formatMessage(tFeedback.uploading)}
+
+
+ +
+
+ +
+ {intl.formatMessage(tNote.newPreview)} +
+
+
+
+
+
+ + +
+ + {(user, index) => ( + } + statNumber={search?.scores[user.pubkey]} + statLabel={intl.formatMessage(tSearch.followers)} + onClick={() => selectUser(user)} + highlighted={highlightedUser() === index()} + /> + )} + +
+
+ + emojiSearchLimit}> +
+ + {(emoji, index) => ( + + )} + +
+
+ +
+
+ + +
+ + +
+
+ ) +} + +export default EditBox; diff --git a/src/components/NewNote/NewNote.module.scss b/src/components/NewNote/NewNote.module.scss new file mode 100644 index 0000000..57a2ed5 --- /dev/null +++ b/src/components/NewNote/NewNote.module.scss @@ -0,0 +1,146 @@ +.newNoteHolder { + min-height: 100px; + background-color: var(--background-site); + padding-top: 10px; +} + +.holderBottomBorder { + width: 100%; + height: 4px; + display: flex; + justify-content: space-between; + + .rightCorner { + display: inline-block; + width: 4px; + height: 4px; + + background-color: var(--background-site); + -webkit-mask: url(../../assets/icons/corner_right.svg) no-repeat center; + mask: url(../../assets/icons/corner_right.svg) no-repeat center; + } + .leftCorner { + display: inline-block; + width: 4px; + height: 4px; + + background-color: var(--background-site); + -webkit-mask: url(../../assets/icons/corner_left.svg) no-repeat center; + mask: url(../../assets/icons/corner_left.svg) no-repeat center; + } +} + +.newNoteBorder { + width: 100%; + height: 100%; + padding: 1px; + background: var(--brand-gradient); + border-radius: 6px; + display: block; + position: relative; +} + +.newNote { + width: 100%; + height: 100%; + min-height: 122px; + font-size: 18px; + line-height: 20px; + margin: 0px; + border-radius: 6px; + border: none; + color: var(--text-tertiary); + background-color: var(--background-site); + padding-bottom: 11px; + display: grid; + grid-template-columns: 92px 1fr; + + .leftSide { + padding: 20px; + } + + + // textarea { + // width: calc(100% - 36px); + // margin: 18px 18px 36px 18px; + // padding: 0px; + // box-sizing: padding-box; + // border: none; + // border-radius: 0px; + // background-color: unset; + // font-size: 16px; + // line-height: 20px; + // font-weight: 400; + // color: var(--text-primary); + // resize: none; + + // &:focus { + // border: none; + // outline: none; + // } + // } +} + +.controls { + position: absolute; + bottom: 0px; + right: 15px; + display: flex; + align-items: center; + >button { + width: 80px; + height: 28px; + margin: 0px 0px 11px 8px; + } +} + +.primaryButton { + border: none; + border-radius: 6px; + margin: 0px 8px; + padding: 0px; + font-size: 14px; + line-height: 20px; + font-weight: 700; + background: var(--brand-gradient-vertical); + color: var(--text-primary); + >span { + opacity: 0.75; + } +} + + +.secondaryButton { + border: none; + border-radius: 6px; + padding: 1px; + font-size: 14px; + line-height: 20px; + font-weight: 700; + background: var(--brand-gradient-vertical); + color: var(--text-tertiary-2); + >div { + width: 100%; + height: 100%; + vertical-align: middle; + border-radius: 6px; + background-color: var(--background-card); + display: flex; + flex-direction: column; + align-items: center; + justify-content: center; + } +} + +.searchSuggestions { + width: 300px; + background-color: var(--background-site); + border: 1px solid var(--text-tertiary-2); + // box-shadow: 0px 4px 12px rgba(0, 0, 0, 0.8); + border-radius: 4px; + + position: absolute; + top: 0px; + left: 0px; + z-index: 50; +} diff --git a/src/components/NewNote/NewNote.tsx b/src/components/NewNote/NewNote.tsx new file mode 100644 index 0000000..81e9c75 --- /dev/null +++ b/src/components/NewNote/NewNote.tsx @@ -0,0 +1,41 @@ +import { Component } from "solid-js"; +import { useAccountContext } from "../../contexts/AccountContext"; +import Avatar from "../Avatar/Avatar"; +import EditBox from "./EditBox/EditBox"; +import styles from "./NewNote.module.scss"; + +const NewNote: Component = () => { + + const account = useAccountContext(); + + const activeUser = () => account?.activeUser; + + return ( + <> +
+
+
+
+ +
+
+ +
+
+
+
+
+
+
+
+ + ) +} + +export default NewNote; diff --git a/src/components/NostrStats/NostrStats.module.scss b/src/components/NostrStats/NostrStats.module.scss new file mode 100644 index 0000000..d0d93e2 --- /dev/null +++ b/src/components/NostrStats/NostrStats.module.scss @@ -0,0 +1,33 @@ + +.statsCaption { + margin-bottom: 32px; + font-weight: 800; + font-size: 18px; + line-height: 20px; + text-transform: uppercase; + color: var(--text-secondary); +} +.netstats { + display: grid; + grid-template-columns: 1fr 1fr 1fr 1fr; + grid-template-rows: 1fr 1fr; + grid-column-gap: 24px; + grid-row-gap: 24px; + + .netstat { + .number { + font-size: 24px; + line-height: 28px; + font-weight: 300; + color: var(--text-primary); + } + + .label { + font-size: 16px; + line-height: 20px; + font-weight: 300; + color: var(--text-tertiary); + text-transform: lowercase; + } + } +} diff --git a/src/components/NostrStats/NostrStats.tsx b/src/components/NostrStats/NostrStats.tsx new file mode 100644 index 0000000..00dfd85 --- /dev/null +++ b/src/components/NostrStats/NostrStats.tsx @@ -0,0 +1,45 @@ +import { useIntl } from "@cookbook/solid-intl"; +import { Component } from "solid-js"; +import { PrimalNetStats } from "../../types/primal"; +import styles from "./NostrStats.module.scss"; +import { explore as t } from '../../translations'; + +const NostrStats: Component<{ stats: PrimalNetStats }> = (props) => { + + const intl = useIntl(); + + const statDisplay = ( + stat: number | string | undefined, + key: string, + ) => { + // @ts-ignore Record find entry by key + const label = t.statDisplay[key] || ''; + + return ( +
+
+ {stat?.toLocaleString()} +
+
+ {intl.formatMessage(label)} +
+
+ ); + }; + + + return ( +
+ {statDisplay(props.stats.users, 'users')} + {statDisplay(props.stats.pubkeys, 'pubkeys')} + {statDisplay(props.stats.zaps, 'users')} + {statDisplay((props.stats.satszapped /100000000).toFixed(8), 'btcZapped')} + {statDisplay(props.stats.pubnotes, 'pubnotes')} + {statDisplay(props.stats.reposts, 'reposts')} + {statDisplay(props.stats.reactions, 'reactions')} + {statDisplay(props.stats.any, 'any')} +
+ ) +} + +export default NostrStats; diff --git a/src/components/Note/MentionedUserLink/MentionedUserLink.module.scss b/src/components/Note/MentionedUserLink/MentionedUserLink.module.scss new file mode 100644 index 0000000..8208ede --- /dev/null +++ b/src/components/Note/MentionedUserLink/MentionedUserLink.module.scss @@ -0,0 +1,39 @@ +.userMention { + position: relative; + .userPreview { + display: none; + } + + &:hover { + .userPreview { + position: absolute; + left: 0; + width: 220px; + z-index: var(--z-index-floater); + background-color: var(--background-input); + display: flex; + padding: 4px; + border: 1px solid var(--text-tertiary-2); + border-radius: 8px; + color: var(--text-secondary-2); + font-size: 14px; + line-height: 16px; + font-weight: 400; + margin-top: 1px; + > div { + display: flex; + flex-direction: column; + justify-content: center; + margin-left: 4px; + + > div { + &.userName { + font-weight: 800; + color: var(--text-primary); + } + padding-block: 2px; + } + } + } + } +} diff --git a/src/components/Note/MentionedUserLink/MentionedUserLink.tsx b/src/components/Note/MentionedUserLink/MentionedUserLink.tsx new file mode 100644 index 0000000..f9c56b8 --- /dev/null +++ b/src/components/Note/MentionedUserLink/MentionedUserLink.tsx @@ -0,0 +1,26 @@ +import { A } from "@solidjs/router"; +import { Component, JSXElement } from "solid-js"; +import { userName, nip05Verification } from "../../../stores/profile"; +import { PrimalUser } from "../../../types/primal"; +import Avatar from "../../Avatar/Avatar"; +import styles from "./MentionedUserLink.module.scss"; + +const MentionedUserLink: Component<{ + user: PrimalUser, + openInNewTab?: boolean, +}> = (props) => { + + const LinkComponent: Component<{ children: JSXElement }> = (p) => { + return props.openInNewTab ? + {p.children} : + {p.children}; + }; + + return ( + + @{userName(props.user)} + + ); +} + +export default MentionedUserLink; diff --git a/src/components/Note/Note.module.scss b/src/components/Note/Note.module.scss new file mode 100644 index 0000000..4d801ae --- /dev/null +++ b/src/components/Note/Note.module.scss @@ -0,0 +1,117 @@ +.post { + background-color: var(--background-card); + display: flex; + flex-direction: column; + // grid-template-columns: 60px 1fr; + // grid-template-rows: 1fr; + // grid-template-areas: "avatar content"; + padding: 0px; + border-radius: 8px; + // pointer-events: none; + + .content { + grid-area: content; + display: flex; + flex-direction: column; + margin-top: 21px; + margin-left: 60px; + // grid-template-columns: 1fr; + // grid-template-rows: 48px 1fr 28px; + // grid-row-gap: 16px; + // grid-template-areas: "header" "message" "footer"; + + + .message { + position: relative; + grid-area: message; + color: var(--text-primary); + word-break: break-word; + font-size: 16px; + line-height: 24px; + width: 100%; + // max-height: 650px; + // overflow: hidden; + margin-bottom: 17px; + + a:hover { + text-decoration: underline; + } + + .messageFade { + position: absolute; + z-index: 1; + top: 610px; + left: 0; + pointer-events: none; + background-image: var(--fade-note-vertical); + width: 100%; + height: 40px; + } + + } + } +} + + +.postLink { + text-decoration: none; + color: unset; + margin: 0px; + padding: 16px 20px; + // background: var(--brand-gradient-vertical); + background-color: var(--background-card); + border-radius: 8px; + display: block; + transition: 0.2s padding; + margin-top: 8px; + >div { + border-radius: 4px; + transition: 0.2s border-radius ease-out; + } + + // &:hover { + // padding-left: 4px; + // transition: 0.2s padding; + // border-radius: 4px; + // >div { + // border-radius: 0px 4px 4px 0px; + // transition: 0.2s border-radius ease-out; + // } + // } +} + +.repostedBy { + padding-bottom: 16px; + display: flex; + >span { + >a { + margin-inline: 5px; + } + color: var(--text-tertiary); + font-size: 16px; + line-height: 16px; + font-weight: 400; + >span { + text-transform: lowercase; + } + } +} + + +@media only screen and (max-width: 720px) { + .postLink { + width: 100vw; + .post { + width: 100%; + // grid-template-columns: 62px 1fr; + margin-left: 0px; + margin-right: 0px; + padding-right: 0px; + .content { + margin-left: 0; + } + } + + } + +} diff --git a/src/components/Note/Note.tsx b/src/components/Note/Note.tsx new file mode 100644 index 0000000..6ee064c --- /dev/null +++ b/src/components/Note/Note.tsx @@ -0,0 +1,71 @@ +import { A } from '@solidjs/router'; +import { Component, Show } from 'solid-js'; +import { PrimalNote } from '../../types/primal'; +import ParsedNote from '../ParsedNote/ParsedNote'; +import NoteFooter from './NoteFooter/NoteFooter'; +import NoteHeader from './NoteHeader/NoteHeader'; + +import styles from './Note.module.scss'; +import { useThreadContext } from '../../contexts/ThreadContext'; +import { useIntl } from '@cookbook/solid-intl'; +import { truncateNpub } from '../../stores/profile'; +import { note as t } from '../../translations'; + +const Note: Component<{ note: PrimalNote }> = (props) => { + + const threadContext = useThreadContext(); + const intl = useIntl(); + + const repost = () => props.note.repost; + + const navToThread = (note: PrimalNote) => { + threadContext?.actions.setPrimaryNote(note); + }; + + const reposterName = () => { + const r = repost(); + + if (!r) { + return ''; + } + + return r.user?.displayName || + r.user?.name || + truncateNpub(r.user.npub); + } + + return ( + navToThread(props.note)} + data-event={props.note.post.id} + data-event-bech32={props.note.post.noteId} + > + +
+
+ +
+ {reposterName()} + + + {intl.formatMessage(t.reposted)} + + +
+ +
+ +
+
+ +
+ +
+
+ + ) +} + +export default Note; diff --git a/src/components/Note/NoteFooter/NoteFooter.module.scss b/src/components/Note/NoteFooter/NoteFooter.module.scss new file mode 100644 index 0000000..6466d40 --- /dev/null +++ b/src/components/Note/NoteFooter/NoteFooter.module.scss @@ -0,0 +1,155 @@ +@mixin statIcon { + width: 16px; + height: 16px; + background-color: var(--text-tertiary-2); +} + +@mixin typeDiv { + display: flex; + min-width: 64px; + align-items: center; +} + +.footer { + display: flex; + position: relative; + + .stat { + // display: grid; + // grid-template-columns: 18px 1fr; + // grid-template-rows: 1fr; + // grid-column-gap: 5px; + font-weight: 400; + font-size: 18px; + line-height: 16px; + align-items: center; + margin: 0px; + padding: 0px; + margin-right: 60px; + border: none; + background-color: unset; + width: auto; + min-width: 64px; + position: relative; + + .likeType { + @include typeDiv; + .icon { + @include statIcon; + -webkit-mask: url(../../../assets/icons/feed_like.svg) no-repeat 0 / 100%; + mask: url(../../../assets/icons/feed_like.svg) no-repeat 0 / 100%; + } + } + + .replyType { + @include typeDiv; + .icon { + @include statIcon; + -webkit-mask: url(../../../assets/icons/feed_reply.svg) no-repeat 0 / 100%; + mask: url(../../../assets/icons/feed_reply.svg) no-repeat 0 / 100%; + } + } + + .repostType { + @include typeDiv; + .icon { + @include statIcon; + -webkit-mask: url(../../../assets/icons/feed_repost.svg) no-repeat 0 / 100%; + mask: url(../../../assets/icons/feed_repost.svg) no-repeat 0 / 100%; + } + } + + .zapType { + @include typeDiv; + .icon { + @include statIcon; + -webkit-mask: url(../../../assets/icons/feed_zap.svg) no-repeat 0 / 100%; + mask: url(../../../assets/icons/feed_zap.svg) no-repeat 0 / 100%; + } + } + + &:hover, &.highlighted { + .zapType { + .statNumber { + color: #FF9F2F; + } + .icon { + background-color: #FF9F2F; + -webkit-mask: url(../../../assets/icons/feed_zap_fill.svg) no-repeat 0 / 100%; + mask: url(../../../assets/icons/feed_zap_fill.svg) no-repeat 0 / 100%; + } + } + + .likeType { + .statNumber { + color: #BC1870; + } + .icon { + background-color: #BC1870; + -webkit-mask: url(../../../assets/icons/feed_like_fill.svg) no-repeat 0 / 100%; + mask: url(../../../assets/icons/feed_like_fill.svg) no-repeat 0 / 100%; + } + } + + .replyType { + .statNumber { + color: #CCCCCC; + } + .icon { + background-color: #CCCCCC; + -webkit-mask: url(../../../assets/icons/feed_reply_fill.svg) no-repeat 0 / 100%; + mask: url(../../../assets/icons/feed_reply_fill.svg) no-repeat 0 / 100%; + } + } + + .repostType { + .statNumber { + color: #66E205; + } + .icon { + background-color: #66E205; + -webkit-mask: url(../../../assets/icons/feed_repost_fill.svg) no-repeat 0 / 100%; + mask: url(../../../assets/icons/feed_repost_fill.svg) no-repeat 0 / 100%; + } + } + } + &:focus { + box-shadow: none; + } + + .statNumber { + text-align: left; + color: var(--text-secondary); + margin-left: 7px; + } + } +} + +.smallZapLottie { + width: 32px; + height: 32px; + position: absolute; + z-index: 20; +} + +.mediumZapLottie { + width: 341px; + height: 91px; + position: absolute; + z-index: 20; +} + + +@media only screen and (max-width: 720px) { + .footer { + width: auto; + display: flex; + align-items: center; + justify-content: space-between; + + .stat{ + min-width: 0px; + margin-right: 0; + } + } +} diff --git a/src/components/Note/NoteFooter/NoteFooter.tsx b/src/components/Note/NoteFooter/NoteFooter.tsx new file mode 100644 index 0000000..dd4f460 --- /dev/null +++ b/src/components/Note/NoteFooter/NoteFooter.tsx @@ -0,0 +1,430 @@ +import { Component, createEffect, createSignal, onMount, Show } from 'solid-js'; +import { PrimalNote } from '../../../types/primal'; +import { sendRepost } from '../../../lib/notes'; + +import styles from './NoteFooter.module.scss'; +import { useAccountContext } from '../../../contexts/AccountContext'; +import { useToastContext } from '../../Toaster/Toaster'; +import { useIntl } from '@cookbook/solid-intl'; + +import { truncateNumber } from '../../../lib/notifications'; +import { canUserReceiveZaps, zapNote } from '../../../lib/zap'; +import CustomZap from '../../CustomZap/CustomZap'; +import { useSettingsContext } from '../../../contexts/SettingsContext'; + +import zapSM from '../../../assets/lottie/zap_sm.json'; +import zapMD from '../../../assets/lottie/zap_md.json'; +import { medZapLimit } from '../../../constants'; +import { toast as t } from '../../../translations'; + +const NoteFooter: Component<{ note: PrimalNote}> = (props) => { + + const account = useAccountContext(); + const toast = useToastContext(); + const intl = useIntl(); + const settings = useSettingsContext(); + + let smallZapAnimation: HTMLElement | undefined; + let medZapAnimation: HTMLElement | undefined; + + const [liked, setLiked] = createSignal(props.note.post.noteActions.liked); + const [zapped, setZapped] = createSignal(props.note.post.noteActions.zapped); + const [replied, setReplied] = createSignal(props.note.post.noteActions.replied); + const [reposted, setReposted] = createSignal(props.note.post.noteActions.reposted); + + const [likes, setLikes] = createSignal(props.note.post.likes); + const [reposts, setReposts] = createSignal(props.note.post.reposts); + const [replies, setReplies] = createSignal(props.note.post.replies); + const [zaps, setZaps] = createSignal(props.note.post.satszapped); + + let footerDiv: HTMLDivElement | undefined; + + const doRepost = async (e: MouseEvent) => { + e.preventDefault(); + e.stopPropagation(); + + if (!account) { + return; + } + + if (Object.keys(account.relaySettings).length === 0) { + toast?.sendWarning( + intl.formatMessage(t.noRelays), + ); + return; + } + + if (account.relays.length === 0) { + toast?.sendWarning( + intl.formatMessage(t.noRelaysConnected), + ); + return; + } + + const success = await sendRepost(props.note, account.relays); + + if (success) { + setReposts(reposts() + 1); + setReposted(true); + toast?.sendSuccess( + intl.formatMessage(t.repostSuccess), + ); + } + else { + toast?.sendWarning( + intl.formatMessage(t.repostFailed), + ); + } + }; + + const doReply = () => {}; + + const doLike = async (e: MouseEvent) => { + e.preventDefault(); + e.stopPropagation(); + + if (!account) { + return; + } + + if (Object.keys(account.relaySettings).length === 0) { + toast?.sendWarning( + intl.formatMessage(noRelaysMessage), + ); + return; + } + + if (account.relays.length === 0) { + toast?.sendWarning( + intl.formatMessage(noRelayConnectedMessage), + ); + return; + } + + const success = await account.actions.addLike(props.note); + + if (success) { + setLikes(likes() + 1); + setLiked(true); + } + }; + + let quickZapDelay = 0; + const [isCustomZap, setIsCustomZap] = createSignal(false); + const [isZapping, setIsZapping] = createSignal(false); + + const startZap = (e: MouseEvent | TouchEvent) => { + e.preventDefault(); + e.stopPropagation(); + + if (!account?.hasPublicKey()) { + toast?.sendWarning( + intl.formatMessage(t.zapAsGuest), + ); + setIsZapping(false); + return; + } + + if (Object.keys(account.relaySettings).length === 0) { + toast?.sendWarning( + intl.formatMessage(noRelaysMessage), + ); + return; + } + + if (account.relays.length === 0) { + toast?.sendWarning( + intl.formatMessage(noRelayConnectedMessage), + ); + return; + } + + if (!canUserReceiveZaps(props.note.user)) { + toast?.sendWarning( + intl.formatMessage(t.zapUnavailable), + ); + setIsZapping(false); + return; + } + + quickZapDelay = setTimeout(() => { + setIsCustomZap(true); + setIsZapping(true); + }, 500); + }; + + const commitZap = (e: MouseEvent | TouchEvent) => { + e.preventDefault(); + e.stopPropagation(); + + clearTimeout(quickZapDelay); + + if (!account?.hasPublicKey() || account.relays.length === 0 || !canUserReceiveZaps(props.note.user)) { + return; + } + + if (!isCustomZap()) { + doQuickZap(); + } + }; + + const [zappedNow, setZappedNow] = createSignal(false); + const [zappedAmount, setZappedAmount] = createSignal(0); + + const animateSmallZap = () => { + setTimeout(() => { + setHideZapIcon(true); + + if (!smallZapAnimation) { + return; + } + + const newLeft = 116; + const newTop = -8; + + smallZapAnimation.style.left = `${newLeft}px`; + smallZapAnimation.style.top = `${newTop}px`; + + const onAnimDone = () => { + // setIsZapping(true); + setShowSmallZapAnim(false); + setHideZapIcon(false); + smallZapAnimation?.removeEventListener('complete', onAnimDone); + } + + smallZapAnimation.addEventListener('complete', onAnimDone); + + try { + // @ts-ignore + smallZapAnimation.seek(0); + // @ts-ignore + smallZapAnimation.play(); + } catch (e) { + console.warn('Failed to animte zap:', e); + onAnimDone(); + } + }, 10); + }; + + const animateMedZap = () => { + setTimeout(() => { + setHideZapIcon(true); + + if (!medZapAnimation) { + return; + } + + const newLeft = 20; + const newTop = -35; + + medZapAnimation.style.left = `${newLeft}px`; + medZapAnimation.style.top = `${newTop}px`; + + const onAnimDone = () => { + // setIsZapping(true); + setShowMedZapAnim(false); + setHideZapIcon(false); + medZapAnimation?.removeEventListener('complete', onAnimDone); + } + + medZapAnimation.addEventListener('complete', onAnimDone); + + try { + // @ts-ignore + medZapAnimation.seek(0); + // @ts-ignore + medZapAnimation.play(); + } catch (e) { + console.warn('Failed to animte zap:', e); + onAnimDone(); + } + }, 10); + }; + + + const animateZap = () => { + if (zappedAmount() > medZapLimit) { + setShowMedZapAnim(true); + animateMedZap(); + } + else { + setShowSmallZapAnim(true); + animateSmallZap(); + } + + } + + const doQuickZap = async () => { + + if (account?.hasPublicKey()) { + setZappedAmount(() => settings?.defaultZapAmount || 0); + setZappedNow(true); + animateZap(); + const success = await zapNote(props.note, account.publicKey, settings?.defaultZapAmount || 10, '', account.relays); + setIsZapping(false); + + if (success) { + return; + } + + setZappedAmount(() => -(settings?.defaultZapAmount || 0)); + setZappedNow(true); + setZapped(props.note.post.noteActions.zapped); + } + } + + const buttonTypeClasses: Record = { + zap: styles.zapType, + like: styles.likeType, + reply: styles.replyType, + repost: styles.repostType, + }; + + const actionButton = (opts: { + type: 'zap' | 'like' | 'reply' | 'repost', + disabled?: boolean, + highlighted?: boolean, + onClick?: (e: MouseEvent) => void, + onMouseDown?: (e: MouseEvent) => void, + onMouseUp?: (e: MouseEvent) => void, + onTouchStart?: (e: TouchEvent) => void, + onTouchEnd?: (e: TouchEvent) => void, + label: string | number, + hidden?: boolean, + title?: string, + }) => { + + return ( + + ); + }; + + createEffect(() => { + + if (zappedNow()) { + setZaps((z) => z + zappedAmount()); + setZapped(true); + setZappedNow(false); + } + + }) + + const [showSmallZapAnim, setShowSmallZapAnim] = createSignal(false); + const [showMedZapAnim, setShowMedZapAnim] = createSignal(false); + const [hideZapIcon, setHideZapIcon] = createSignal(false); + + return ( +
+ + + + + + + + + {actionButton({ + onClick: doReply, + type: 'reply', + highlighted: replied(), + label: replies() === 0 ? '' : truncateNumber(replies(), 2), + title: replies().toLocaleString(), + })} + + {actionButton({ + onClick: (e: MouseEvent) => e.preventDefault(), + onMouseDown: startZap, + onMouseUp: commitZap, + onTouchStart: startZap, + onTouchEnd: commitZap, + type: 'zap', + highlighted: zapped() || isZapping(), + label: zaps() === 0 ? '' : truncateNumber(zaps(), 2), + hidden: hideZapIcon(), + title: zaps().toLocaleString(), + })} + + {actionButton({ + onClick: doLike, + type: 'like', + highlighted: liked(), + label: likes() === 0 ? '' : truncateNumber(likes(), 2), + title: likes().toLocaleString(), + })} + + {actionButton({ + onClick: doRepost, + type: 'repost', + highlighted: reposted(), + label: reposts() === 0 ? '' : truncateNumber(reposts(), 2), + title: reposts().toLocaleString(), + })} + + { + setIsCustomZap(false); + setZappedAmount(() => amount || 0); + setZappedNow(true); + setZapped(true); + animateZap(); + }} + onSuccess={(amount) => { + setIsCustomZap(false); + setIsZapping(false); + // setZappedAmount(() => amount || 0); + setZappedNow(false); + // animateZap(); + setShowMedZapAnim(false); + setShowSmallZapAnim(false); + setHideZapIcon(false); + setZapped(true); + }} + onFail={(amount) => { + setZappedAmount(() => -(amount || 0)); + setZappedNow(true); + setIsCustomZap(false); + setIsZapping(false); + setShowMedZapAnim(false); + setShowSmallZapAnim(false); + setHideZapIcon(false); + setZapped(props.note.post.noteActions.zapped); + }} + /> + +
+ ) +} + +export default NoteFooter; diff --git a/src/components/Note/NoteHeader/NoteHeader.module.scss b/src/components/Note/NoteHeader/NoteHeader.module.scss new file mode 100644 index 0000000..150e059 --- /dev/null +++ b/src/components/Note/NoteHeader/NoteHeader.module.scss @@ -0,0 +1,157 @@ +.header { + grid-area: header; + align-items: center; + color: var(--text-tertiary-2); + display: flex; + justify-content: space-between; + width: 100%; + height: 48px; + + .headerInfo { + display: flex; + flex-direction: row; + justify-content: flex-start; + } + + .avatar { + margin-right: 12px; + + >a { + text-decoration: none; + } + + .avatarImg { + width: 52px; + height: 52px; + border-radius: 50%; + } + } + .postInfo { + display: flex; + flex-direction: column; + font-size: 14px; + line-height: 16px; + font-weight: 400; + height: 48px; + justify-content: space-around; + + .verification { + max-width: 470px; + text-overflow: ellipsis; + white-space: nowrap; + overflow: hidden; + + } + .userInfo { + display: flex; + justify-content: flex-start; + overflow: hidden; + display: flex; + align-items: center; + justify-content: flex-start; + .userName { + text-overflow: ellipsis; + white-space: nowrap; + overflow: hidden; + max-width: 360px; + color: var(--text-primary); + } + .time{ + margin: 0px 2px; + min-width: 80px; + &::before { + content: "|"; + padding: 0px 2px; + } + } + } + + } + + + .contextMenu { + position: relative; + width: 16px; + height: 16px; + display: flex; + align-items: center; + text-align: center; + font-weight: bold; + + .contextButton { + height: 16px; + padding: 0; + margin: 0; + background: none; + border: none; + outline: none; + + &:focus { + outline: none; + box-shadow: none; + } + } + } + +} + +.contextIcon { + width: 16px; + height: 16px; + background-color: var(--text-secondary-2); + -webkit-mask: url(../../../assets/icons/context.svg) no-repeat 0 / 100%; + mask: url(../../../assets/icons/context.svg) no-repeat 0 / 100%; + + &:hover { + background-color: var(--text-primary); + } +} + +.contextMenuOptions { + position: absolute; + top: 16px; + right: 0px; + min-width: 160px; + background-color: var(--background-site); + border: solid 1px var(--text-tertiary-2); + border-radius: 4px; + z-index: 20; +} + +.contextOption { + background-color: var(--background-site); + border: none; + font-weight: 400; + font-size: 14px; + line-height: 16px; + color: var(--text-secondary-2); + padding: 10px; + margin: 0px; + + &:hover, &:focus { + background-color: var(--background-input); + } +} + +@media only screen and (max-width: 720px) { + .header { + width: calc(100vw - 160px); + .postInfo { + width: calc(100vw - 110px); + .userInfo { + max-width: calc(100vw - 100px); + overflow: hidden; + .userName { + max-width: calc(100vw - 180px); + } + } + .verification { + max-width: 220px; + text-overflow: ellipsis; + white-space: nowrap; + overflow: hidden; + + } + } + } +} diff --git a/src/components/Note/NoteHeader/NoteHeader.tsx b/src/components/Note/NoteHeader/NoteHeader.tsx new file mode 100644 index 0000000..26e2a3c --- /dev/null +++ b/src/components/Note/NoteHeader/NoteHeader.tsx @@ -0,0 +1,145 @@ +import { Component, createEffect, createSignal, Show } from 'solid-js'; +import { PrimalNote } from '../../../types/primal'; + +import styles from './NoteHeader.module.scss'; +import { date } from '../../../lib/dates'; +import { nip05Verification, truncateNpub } from '../../../stores/profile'; +import { useIntl } from '@cookbook/solid-intl'; +import { useToastContext } from '../../Toaster/Toaster'; +import VerificationCheck from '../../VerificationCheck/VerificationCheck'; +import Avatar from '../../Avatar/Avatar'; +import { A } from '@solidjs/router'; +import { toast as tToast, actions as tActions } from '../../../translations'; + +const NoteHeader: Component<{ note: PrimalNote}> = (props) => { + + const intl = useIntl(); + const toaster = useToastContext(); + + const [showContext, setContext] = createSignal(false); + + const authorName = () => { + return props.note.user?.displayName || + props.note.user?.name || + truncateNpub(props.note.user.npub); + }; + + const openContextMenu = (e: MouseEvent) => { + e.preventDefault(); + setContext(true); + }; + + const copyNostrLink = (e: MouseEvent) => { + e.preventDefault(); + navigator.clipboard.writeText(`nostr:${props.note.post.noteId}`); + setContext(false); + toaster?.sendSuccess(intl.formatMessage(tToast.noteNostrLinkCoppied)); + }; + + const copyPrimalLink = (e: MouseEvent) => { + e.preventDefault(); + navigator.clipboard.writeText(`${window.location.origin}/thread/${props.note.post.noteId}`); + setContext(false); + toaster?.sendSuccess(intl.formatMessage(tToast.notePrimalLinkCoppied)); + }; + + const onClickOutside = (e: MouseEvent) => { + if ( + !document?.getElementById(`note_context_${props.note.post.id}`)?.contains(e.target as Node) + ) { + setContext(false); + } + } + + createEffect(() => { + if (showContext()) { + document.addEventListener('click', onClickOutside); + } + else { + document.removeEventListener('click', onClickOutside); + } + }); + + const isVerifiedByPrimal = () => { + return !!props.note.user.nip05 && + props.note.user.nip05.endsWith('primal.net'); + } + + return ( +
+
+
+ + + +
+
+
+ + + {authorName()} + + + + + + {date(props.note.post?.created_at).label} + +
+ + + + {nip05Verification(props.note.user)} + + +
+
+
+ + +
+ + +
+
+
+
+ ) +} + +export default NoteHeader; diff --git a/src/components/Note/NotePrimary/NotePrimary.module.scss b/src/components/Note/NotePrimary/NotePrimary.module.scss new file mode 100644 index 0000000..d94512f --- /dev/null +++ b/src/components/Note/NotePrimary/NotePrimary.module.scss @@ -0,0 +1,110 @@ +.border { + background: var(--brand-gradient-vertical); + border-radius: 4px 0px 0px 4px; + width: 4px; + position: absolute; + top: 0; + left: 0; + height: 100%; +} + +.post { + position: relative; + background-color: var(--background-card); + margin-top: 0px; + display: flex; + flex-direction: column; + padding: 16px 20px; + // grid-template-columns: 122px 1fr; + // grid-template-rows: 1fr; + // grid-template-areas: "avatar content"; + // padding: 16px 16px 16px 20px; + border-radius: 0px 4px 4px 0px; + + // .avatar { + // grid-area: avatar; + // display: grid; + // grid-template-columns: 1fr; + // grid-template-rows: 80px 1fr; + // grid-row-gap: 6px; + // justify-items: center; + + // >a { + // text-decoration: none; + // } + + // .avatarName { + // width: 100%; + // padding: 0px 6px; + // text-align: center; + // display: inline-block; + // font-size: 16px; + // line-height: 12px; + // font-weight: 400; + // color: var(--text-primary); + // align-self: flex-start; + // text-overflow: ellipsis; + // white-space: nowrap; + // overflow: hidden; + // } + // } + + .content { + // grid-area: content; + // display: grid; + // grid-template-columns: 1fr; + // grid-template-rows: 48px 1fr 28px; + // grid-row-gap: 16px; + // grid-template-areas: "header" "message" "footer"; + grid-area: content; + display: flex; + flex-direction: column; + margin-top: 21px; + margin-left: 60px; + + + .message { + position: relative; + grid-area: message; + color: var(--text-primary); + word-break: break-word; + font-size: 16px; + line-height: 24px; + width: 100%; + // max-height: 650px; + // overflow: hidden; + margin-bottom: 17px; + + a:hover { + text-decoration: underline; + } + + .messageFade { + position: absolute; + z-index: 1; + top: 610px; + left: 0; + pointer-events: none; + background-image: var(--fade-note-vertical); + width: 100%; + height: 40px; + } + + } + } +} + + +@media only screen and (max-width: 720px) { + .post { + width: 100vw; + // grid-template-columns: 62px 1fr; + margin-left: 0px; + margin-right: 0px; + padding-right: 12px; + .content { + margin-left: 0px; + } + } + +} diff --git a/src/components/Note/NotePrimary/NotePrimary.tsx b/src/components/Note/NotePrimary/NotePrimary.tsx new file mode 100644 index 0000000..49577d5 --- /dev/null +++ b/src/components/Note/NotePrimary/NotePrimary.tsx @@ -0,0 +1,33 @@ +import { Component } from 'solid-js'; +import { truncateNpub } from '../../../stores/profile'; +import { PrimalNote } from '../../../types/primal'; +import ParsedNote from '../../ParsedNote/ParsedNote'; +import NoteFooter from '../NoteFooter/NoteFooter'; +import NoteHeader from '../NoteHeader/NoteHeader'; + +import styles from './NotePrimary.module.scss'; + + +const NotePrimary: Component<{ note: PrimalNote }> = (props) => { + + return ( +
+
+ +
+ +
+ +
+ + +
+
+ ) +} + +export default NotePrimary; diff --git a/src/components/Note/NotificationNote/NotificationNote.module.scss b/src/components/Note/NotificationNote/NotificationNote.module.scss new file mode 100644 index 0000000..e5c5c8c --- /dev/null +++ b/src/components/Note/NotificationNote/NotificationNote.module.scss @@ -0,0 +1,101 @@ +.post { + background-color: var(--background-site); + display: grid; + grid-template-columns: 1fr; + grid-template-rows: 1fr; + grid-template-areas: "content"; + padding: 0px 16px 16px 0px; + border-radius: 4px; + // width: 640px; + // pointer-events: none; + + .content { + grid-area: content; + display: flex; + flex-direction: column; + + + .message { + grid-area: message; + color: var(--text-primary); + word-break: break-word; + font-size: 16px; + line-height: 24px; + width: 100%; + + a:hover { + text-decoration: underline; + } + } + + .footer { + margin-top: 12px; + } + } +} + + +.postLink { + text-decoration: none; + color: unset; + margin: 0px; + padding: 0px; + // background: var(--brand-gradient-vertical); + background-color: var(--background-card); + border-radius: 6px; + display: block; + transition: 0.2s padding; + margin-top: 6px; + >div { + border-radius: 4px; + transition: 0.2s border-radius ease-out; + } + + // &:hover { + // padding-left: 4px; + // transition: 0.2s padding; + // border-radius: 4px; + // >div { + // border-radius: 0px 4px 4px 0px; + // transition: 0.2s border-radius ease-out; + // } + // } +} + +.repostedBy { + margin-left: 22px; + padding-top: 16px; + display: flex; + >span { + >a { + margin-inline: 5px; + } + color: var(--text-tertiary); + font-size: 16px; + line-height: 16px; + font-weight: 400; + >span { + text-transform: lowercase; + } + } +} + + +@media only screen and (max-width: 720px) { + .post { + width: calc(100vw - (100vw - 100%)); + grid-template-columns: 62px 1fr; + margin-left: 0px; + margin-right: 0px; + padding-right: 0px; + .content { + .message { + width: 80% !important; + } + } + } + + .postLink { + width: 100% !important; + } +} diff --git a/src/components/Note/NotificationNote/NotificationNote.tsx b/src/components/Note/NotificationNote/NotificationNote.tsx new file mode 100644 index 0000000..9c4c85d --- /dev/null +++ b/src/components/Note/NotificationNote/NotificationNote.tsx @@ -0,0 +1,41 @@ +import { A } from '@solidjs/router'; +import { Component } from 'solid-js'; +import { PrimalNote } from '../../../types/primal'; +import ParsedNote from '../../ParsedNote/ParsedNote'; +import NoteFooter from '../NoteFooter/NoteFooter'; + +import styles from './NotificationNote.module.scss'; +import { useThreadContext } from '../../../contexts/ThreadContext'; + +const Note: Component<{ note: PrimalNote }> = (props) => { + + const threadContext = useThreadContext(); + + const navToThread = (note: PrimalNote) => { + threadContext?.actions.setPrimaryNote(note); + }; + + return ( + navToThread(props.note)} + data-event={props.note.post.id} + data-event-bech32={props.note.post.noteId} + > +
+
+
+ +
+ +
+ +
+
+
+
+ ) +} + +export default Note; diff --git a/src/components/NotificationAvatar/NotificationAvatar.module.scss b/src/components/NotificationAvatar/NotificationAvatar.module.scss new file mode 100644 index 0000000..e8a9913 --- /dev/null +++ b/src/components/NotificationAvatar/NotificationAvatar.module.scss @@ -0,0 +1,165 @@ +@mixin avatar { + position: relative; + + background-color: var(--subtile-devider); + border-radius: 50%; + border: 2px solid var(--text-primary); + color: var(--text-primary); + font-weight: 500; + font-size: 12px; + line-height: 18px; + display: flex; + align-items: center; + justify-content: center; +} + +.verifiedIcon { + position: absolute; + top: 0px; + right: 0px; + width: 15px; + height: 15px; + display: inline-block; + margin: 0px 0px; + background-color: var(--accent-2); + -webkit-mask: url(../../assets/icons/verified.svg) no-repeat 0px / 15px; + mask: url(../../assets/icons/verified.svg) no-repeat 0px / 15px; +} + +@mixin iconBackground { + position: absolute; + right: 0px; + bottom: 0px; + width: 15px; + height: 15px; + background-color: var(--background-site); + border-radius: 50%; +} + +.xxsAvatar { + @include avatar; + width: 24px; + height: 24px; +} + +.xsAvatar { + @include avatar; + width: 36px; + height: 36px; +} + +.vsAvatar { + @include avatar; + width: 42px; + height: 42px; +} + +.smallAvatar { + @include avatar; + width: 48px; + height: 48px; +} + +.midAvatar { + @include avatar; + width: 52px; + height: 52px; +} + +.largeAvatar { + @include avatar; + width: 72px; + height: 72px; +} + +.extraLargeAvatar { + @include avatar; + width: 80px; + height: 80px; +} + +.xxlAvatar { + @include avatar; + width: 142px; + height: 142px; +} + +@mixin missing { + display: grid; + place-items: center; + color: var(--missing-avatar-text); + background-color: var(--subtile-devider); + border-radius: 50%; +} + +.xxsMissing { + @include missing; + width: 24px; + height: 24px; + font-size: 10px; + -webkit-mask: url(../../assets/icons/default_nostrich.svg) no-repeat 0px / 24px; + mask: url(../../assets/icons/default_nostrich.svg) no-repeat 0px / 24px; +} + +.xsMissing { + @include missing; + width: 32px; + height: 32px; + font-size: 10px; + -webkit-mask: url(../../assets/icons/default_nostrich.svg) no-repeat 0px / 32px; + mask: url(../../assets/icons/default_nostrich.svg) no-repeat 0px / 32px; +} + +.vsMissing { + @include missing; + width: 42px; + height: 42px; + font-size: 10px; + -webkit-mask: url(../../assets/icons/default_nostrich.svg) no-repeat 0px / 42px; + mask: url(../../assets/icons/default_nostrich.svg) no-repeat 0px / 42px; +} + +.smallMissing { + @include missing; + width: 48px; + height: 48px; + font-size: 12px; + -webkit-mask: url(../../assets/icons/default_nostrich.svg) no-repeat 0px / 48px; + mask: url(../../assets/icons/default_nostrich.svg) no-repeat 0px / 48px; +} + +.midMissing { + @include missing; + width: 52px; + height: 52px; + font-size: 16px; + -webkit-mask: url(../../assets/icons/default_nostrich.svg) no-repeat 0px / 52px; + mask: url(../../assets/icons/default_nostrich.svg) no-repeat 0px / 52px; +} + +.largeMissing { + @include missing; + width: 72px; + height: 72px; + font-size: 18px; + -webkit-mask: url(../../assets/icons/default_nostrich.svg) no-repeat 0px / 72px; + mask: url(../../assets/icons/default_nostrich.svg) no-repeat 0px / 72px; +} + +.extraLargeMissing { + @include missing; + width: 80px; + height: 80px; + font-size: 20px; + -webkit-mask: url(../../assets/icons/default_nostrich.svg) no-repeat 0px / 80px; + mask: url(../../assets/icons/default_nostrich.svg) no-repeat 0px / 80px; +} + +.xxlMissing { + @include missing; + width: 142px; + height: 142px; + font-size: 20px; + -webkit-mask: url(../../assets/icons/default_nostrich.svg) no-repeat 0px / 142px; + mask: url(../../assets/icons/default_nostrich.svg) no-repeat 0px / 142px; +} diff --git a/src/components/NotificationAvatar/NotificationAvatar.tsx b/src/components/NotificationAvatar/NotificationAvatar.tsx new file mode 100644 index 0000000..e7323ac --- /dev/null +++ b/src/components/NotificationAvatar/NotificationAvatar.tsx @@ -0,0 +1,62 @@ +import { Component, Show } from 'solid-js'; +import defaultAvatar from '../../assets/icons/default_nostrich.svg'; + +import styles from './NotificationAvatar.module.scss'; + +const NotificationAvatar: Component<{ + number: number | undefined, + size?: "xxs" | "xs" | "vs" | "sm" | "md" | "lg" | "xl" | "xxl", + verified?: string +}> = (props) => { + + const selectedSize = props.size || 'sm'; + + const avatarClass = { + xxs: styles.xxsAvatar, + xs: styles.xsAvatar, + vs: styles.vsAvatar, + sm: styles.smallAvatar, + md: styles.midAvatar, + lg: styles.largeAvatar, + xl: styles.extraLargeAvatar, + xxl: styles.xxlAvatar, + }; + + const missingClass = { + xxs: styles.xxsMissing, + xs: styles.xsMissing, + vs: styles.vsMissing, + sm: styles.smallMissing, + md: styles.midMissing, + lg: styles.largeMissing, + xl: styles.extraLargeMissing, + xxl: styles.xxlMissing, + }; + + const imgError = (event: any) => { + const image = event.target; + image.onerror = ""; + image.src = defaultAvatar; + return true; + } + + return ( +
+
+ } + > + +{props.number} + + +
+
+
+
+ + ) +} + +export default NotificationAvatar; diff --git a/src/components/Notifications/NotificationItem.module.scss b/src/components/Notifications/NotificationItem.module.scss new file mode 100644 index 0000000..3481231 --- /dev/null +++ b/src/components/Notifications/NotificationItem.module.scss @@ -0,0 +1,93 @@ +.notifItem { + display: grid; + grid-template-columns: 44px 1fr; + + padding-top: 12px; + padding-bottom: 17px; + border-bottom: 1px solid var(--subtile-devider); + + .notifType { + display: flex; + flex-direction: column; + align-items: center; + justify-content: flex-start; + padding-block: 6px; + } + + .iconInfo { + color: #FFA02F; + font-size: 14px; + line-height: 16px; + font-weight: 700; + margin-top: 6px; + } + + .notifContent { + display: flex; + flex-direction: column; + + .avatars { + display: flex; + padding-bottom: 8px; + .avatar { + border: solid 2px var(--text-primary); + border-radius: 50%; + width: 36px; + transition: margin-right 0.2s; + margin-right: -16px; + + &:hover { + transition: margin-right 0.2s; + margin-right: 5px; + } + } + } + + .description { + display: flex; + align-items: baseline; + padding-bottom: 8px; + .firstUser { + display: flex; + align-items: center; + font-weight: 800; + font-size: 16px; + line-height: 18px; + color: var(--text-secondary); + } + + .restUsers { + margin-left: 4px; + font-weight: 400; + font-size: 16px; + line-height: 18px; + color: var(--text-secondary); + } + } + + .reference { + font-weight: 400; + font-size: 16px; + line-height: 22px; + color: var(--text-tertiary); + } + } +} + +.firstUserName { + max-width: 200px; + text-overflow: ellipsis; + white-space: nowrap; + overflow: hidden; +} + +.verifiedIcon { + width: 14px; + height: 14px; + min-width: 14px; + display: inline-block; + margin-left: 2px; + background-color: var(--text-tertiary-2); + -webkit-mask: url(../../assets/icons/verified.svg) no-repeat 0 / 14px; + mask: url(../../../assets/icons/verified.svg) no-repeat 0 / 14px; +} diff --git a/src/components/Notifications/NotificationItem.tsx b/src/components/Notifications/NotificationItem.tsx new file mode 100644 index 0000000..2f61e20 --- /dev/null +++ b/src/components/Notifications/NotificationItem.tsx @@ -0,0 +1,195 @@ +import { useIntl } from '@cookbook/solid-intl'; +import { A } from '@solidjs/router'; +import { Component, createEffect, createMemo, createSignal, For, Show } from 'solid-js'; +import { NotificationType } from '../../constants'; +import { trimVerification } from '../../lib/profile'; +import { truncateNpub, userName } from '../../stores/profile'; +import { PrimalNote, PrimalNotifUser } from '../../types/primal'; +import Avatar from '../Avatar/Avatar'; + +import styles from './NotificationItem.module.scss'; + +import userFollow from '../../assets/icons/notifications/user_followed.svg'; +import userUnFollow from '../../assets/icons/notifications/user_unfollowed.svg'; + +import postZapped from '../../assets/icons/notifications/post_zapped.svg'; +import postLiked from '../../assets/icons/notifications/post_liked.svg'; +import postReposted from '../../assets/icons/notifications/post_reposted.svg'; +import postReplied from '../../assets/icons/notifications/post_replied.svg'; + +import mention from '../../assets/icons/notifications/mention.svg'; +import mentionedPost from '../../assets/icons/notifications/mentioned_post.svg'; + +import mentionZapped from '../../assets/icons/notifications/mention_zapped.svg'; +import mentionLiked from '../../assets/icons/notifications/mention_liked.svg'; +import mentionReposted from '../../assets/icons/notifications/mention_reposted.svg'; +import mentionReplied from '../../assets/icons/notifications/mention_replied.svg'; + +import mentionedPostZapped from '../../assets/icons/notifications/mentioned_post_zapped.svg'; +import mentionedPostLiked from '../../assets/icons/notifications/mentioned_post_liked.svg'; +import mentionedPostReposted from '../../assets/icons/notifications/mentioned_post_reposted.svg'; +import mentionedPostReplied from '../../assets/icons/notifications/mentioned_post_replied.svg'; +import NotificationNote from '../Note/NotificationNote/NotificationNote'; +import NotificationAvatar from '../NotificationAvatar/NotificationAvatar'; +import { notificationsNew as t } from '../../translations'; + +const typeIcons: Record = { + [NotificationType.NEW_USER_FOLLOWED_YOU]: userFollow, + [NotificationType.USER_UNFOLLOWED_YOU]: userUnFollow, + + [NotificationType.YOUR_POST_WAS_ZAPPED]: postZapped, + [NotificationType.YOUR_POST_WAS_LIKED]: postLiked, + [NotificationType.YOUR_POST_WAS_REPOSTED]: postReposted, + [NotificationType.YOUR_POST_WAS_REPLIED_TO]: postReplied, + + [NotificationType.YOU_WERE_MENTIONED_IN_POST]: mention, + [NotificationType.YOUR_POST_WAS_MENTIONED_IN_POST]: mentionedPost, + + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_ZAPPED]: mentionZapped, + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_LIKED]: mentionLiked, + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_REPOSTED]: mentionReposted, + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_REPLIED_TO]: mentionReplied, + + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_ZAPPED]: mentionedPostZapped, + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_LIKED]: mentionedPostLiked, + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPOSTED]:mentionedPostReposted, + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPLIED_TO]: mentionedPostReplied, + +} + +type NotificationItemProps = { + type: NotificationType, + users?: PrimalNotifUser[], + note?: PrimalNote, + iconInfo?: string, + iconTooltip?: string, +}; + +const uniqueifyUsers = (users: PrimalNotifUser[]) => { + return users.reduce((acc, u) => { + const found = acc.find(a => a.id === u.id); + return found ? acc : [...acc, u]; + }, []); +} + +const avatarDisplayLimit = 12; + +const NotificationItem: Component = (props) => { + + const intl = useIntl(); + + const [typeIcon, setTypeIcon] = createSignal(''); + + const sortedUsers = createMemo(() => { + if (!props.users || props.users.length === 0) { + return []; + } + + const users = uniqueifyUsers(props.users); + + return users.sort((a, b) => b.followers_count - a.followers_count); + }); + + const displayedUsers = createMemo(() => { + const limited = sortedUsers().slice(0, avatarDisplayLimit); + + return limited; + }); + + const numberOfUsers = createMemo(() => sortedUsers().length); + + const remainingUsers = createMemo(() => { + const remainder = numberOfUsers() - displayedUsers().length; + + return remainder > 99 ? 99 : remainder; + }); + + + const firstUserName = createMemo(() => { + const firstUser = sortedUsers()[0]; + + if (!firstUser) { + return ''; + } + + return firstUser.displayName || + firstUser.name || + truncateNpub(firstUser.npub); + }); + + const firstUserVerification = createMemo(() => { + const firstUser = sortedUsers()[0]; + + if (!firstUser || !firstUser.nip05) { + return null; + } + + return trimVerification(firstUser.nip05); + + }); + + const typeDescription = () => { + + return intl.formatMessage(t[props.type], { + number: numberOfUsers() - 1, + }); + + } + + createEffect(() => { + setTypeIcon(typeIcons[props.type]) + }); + + return ( +
+
+ notification icon +
+ {props.iconInfo} +
+
+
+
+ 0}> + + {(user) => ( + + + + )} + + + avatarDisplayLimit - 1}> + + +
+
+
+ {firstUserName()} + + + +
+
{typeDescription()}
+
+ +
+ + + +
+
+
+
+ ); +} + +export default NotificationItem; diff --git a/src/components/Notifications/NotificationItem2.tsx b/src/components/Notifications/NotificationItem2.tsx new file mode 100644 index 0000000..16752e4 --- /dev/null +++ b/src/components/Notifications/NotificationItem2.tsx @@ -0,0 +1,161 @@ +import { useIntl } from '@cookbook/solid-intl'; +import { A } from '@solidjs/router'; +import { Component, createEffect, createSignal, Show } from 'solid-js'; +import { NotificationType, notificationTypeNoteProps, notificationTypeUserProps } from '../../constants'; +import { trimVerification } from '../../lib/profile'; +import { userName } from '../../stores/profile'; +import { PrimalNote, PrimalNotification, PrimalUser } from '../../types/primal'; +import Avatar from '../Avatar/Avatar'; + +import styles from './NotificationItem.module.scss'; + +import userFollow from '../../assets/icons/notifications/user_followed.svg'; +import userUnFollow from '../../assets/icons/notifications/user_unfollowed.svg'; + +import postZapped from '../../assets/icons/notifications/post_zapped.svg'; +import postLiked from '../../assets/icons/notifications/post_liked.svg'; +import postReposted from '../../assets/icons/notifications/post_reposted.svg'; +import postReplied from '../../assets/icons/notifications/post_replied.svg'; + +import mention from '../../assets/icons/notifications/mention.svg'; +import mentionedPost from '../../assets/icons/notifications/mentioned_post.svg'; + +import mentionZapped from '../../assets/icons/notifications/mention_zapped.svg'; +import mentionLiked from '../../assets/icons/notifications/mention_liked.svg'; +import mentionReposted from '../../assets/icons/notifications/mention_reposted.svg'; +import mentionReplied from '../../assets/icons/notifications/mention_replied.svg'; + +import mentionedPostZapped from '../../assets/icons/notifications/mentioned_post_zapped.svg'; +import mentionedPostLiked from '../../assets/icons/notifications/mentioned_post_liked.svg'; +import mentionedPostReposted from '../../assets/icons/notifications/mentioned_post_reposted.svg'; +import mentionedPostReplied from '../../assets/icons/notifications/mentioned_post_replied.svg'; +import NotificationNote from '../Note/NotificationNote/NotificationNote'; +import { truncateNumber } from '../../lib/notifications'; +import { notificationsOld as t } from '../../translations'; + +const typeIcons: Record = { + [NotificationType.NEW_USER_FOLLOWED_YOU]: userFollow, + [NotificationType.USER_UNFOLLOWED_YOU]: userUnFollow, + + [NotificationType.YOUR_POST_WAS_ZAPPED]: postZapped, + [NotificationType.YOUR_POST_WAS_LIKED]: postLiked, + [NotificationType.YOUR_POST_WAS_REPOSTED]: postReposted, + [NotificationType.YOUR_POST_WAS_REPLIED_TO]: postReplied, + + [NotificationType.YOU_WERE_MENTIONED_IN_POST]: mention, + [NotificationType.YOUR_POST_WAS_MENTIONED_IN_POST]: mentionedPost, + + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_ZAPPED]: mentionZapped, + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_LIKED]: mentionLiked, + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_REPOSTED]: mentionReposted, + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_REPLIED_TO]: mentionReplied, + + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_ZAPPED]: mentionedPostZapped, + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_LIKED]: mentionedPostLiked, + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPOSTED]:mentionedPostReposted, + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPLIED_TO]: mentionedPostReplied, + +} + +type NotificationItemProps = { + notes: PrimalNote[], + users: Record, + userStats: Record, + notification: PrimalNotification, +}; + +const NotificationItem2: Component = (props) => { + + const intl = useIntl(); + + const [typeIcon, setTypeIcon] = createSignal(''); + + const type = () => props.notification.type + + const note = () => { + const prop = notificationTypeNoteProps[type()]; + // @ts-ignore + const id = props.notification[prop]; + return props.notes.find(n => n.post.id === id) + }; + + const user = () => { + const prop = notificationTypeUserProps[type()]; + // @ts-ignore + const id = props.notification[prop]; + return props.users[`${id}`]; + }; + + const typeDescription = () => { + return intl.formatMessage(t[type()]); + + } + + createEffect(() => { + setTypeIcon(typeIcons[type()]) + }); + + + const isReply = () => { + return [ + NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPLIED_TO, + NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_REPLIED_TO, + NotificationType.YOUR_POST_WAS_REPLIED_TO, + ].includes(type()) + }; + + const isZapType = () => { + return [ + NotificationType.YOUR_POST_WAS_ZAPPED, + NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_ZAPPED, + NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_ZAPPED, + ].includes(type()) + }; + + + return ( +
+
+ notification icon + +
+ {truncateNumber(props.notification.satszapped || 0)} +
+
+
+
+
+ + + +
+
+
+ {userName(user())} + + + +
+
{typeDescription()}
+
+ +
+ + + +
+
+
+
+ ); +} + +export default NotificationItem2; diff --git a/src/components/NotificatiosSidebar/NotificationsSidebar.module.scss b/src/components/NotificatiosSidebar/NotificationsSidebar.module.scss new file mode 100644 index 0000000..25cdcd0 --- /dev/null +++ b/src/components/NotificatiosSidebar/NotificationsSidebar.module.scss @@ -0,0 +1,90 @@ +.sidebarHeading { + font-size: 18px; + font-weight: 800; + line-height: 22px; + color: var(--text-secondary-2); + text-transform: uppercase; + margin-bottom: 24px; +} + +.sidebarTitle { + font-weight: 800; + font-size: 18px; + line-height: 22px; + color: var(--text-secondary); +} + +.sidebarEmpty { + color: var(--text-secondary); + font-weight: 400; + font-size: 18px; + line-height: 20px; + text-transform: lowercase; +} + +.sidebarItems { + margin-bottom: 20px; + + .sidebarItem { + font-weight: 400; + font-size: 16px; + line-height: 22px; + color: var(--text-secondary); + + .itemAmount { + display: inline-block; + font-weight: 600; + font-size: 16px; + line-height: 22px; + color: var(--text-primary); + margin-right: 4px; + } + } +} + +.category { + display: flex; +} + +.categoryIcon { + display: flex; + padding-top: 3px; + width: 26px; + justify-content: center; +} + +@mixin statIcon { + width: 16px; + height: 16px; + background-color: var(--text-secondary); +} + +.followIcon { + @include statIcon; + -webkit-mask: url(../../assets/icons/notifications/follows.svg) no-repeat 0px / 16px; + mask: url(../../assets/icons/notifications/follows.svg) no-repeat 0px / 16px; +} + +.mentionIcon { + @include statIcon; + -webkit-mask: url(../../assets/icons/notifications/at.svg) no-repeat 0px / 16px; + mask: url(../../assets/icons/notifications/at.svg) no-repeat 0px / 16px; +} + +.zapIcon { + @include statIcon; + -webkit-mask: url(../../assets/icons/zaps_filled.svg) no-repeat 0px / 16px; + mask: url(../../assets/icons/zaps_filled.svg) no-repeat 0px / 16px; +} + +.activityIcon { + @include statIcon; + -webkit-mask: url(../../assets/icons/notifications/post_replied.svg) no-repeat 0px / 18px; + mask: url(../../assets/icons/notifications/post_replied.svg) no-repeat 0px / 18px; +} + +.contextIcon { + @include statIcon; + -webkit-mask: url(../../assets/icons/context.svg) no-repeat center; + mask: url(../../assets/icons/context.svg) no-repeat center; +} diff --git a/src/components/NotificatiosSidebar/NotificationsSidebar.tsx b/src/components/NotificatiosSidebar/NotificationsSidebar.tsx new file mode 100644 index 0000000..59cc8e1 --- /dev/null +++ b/src/components/NotificatiosSidebar/NotificationsSidebar.tsx @@ -0,0 +1,375 @@ +import { useIntl } from '@cookbook/solid-intl'; +import { Component, For, Show } from 'solid-js'; +import { NotificationType } from '../../constants'; +import { truncateNumber } from '../../lib/notifications'; +import { PrimalNotification, PrimalNotifUser, SortedNotifications } from '../../types/primal'; +import { notificationsSidebar as t } from '../../translations'; + +import styles from './NotificationsSidebar.module.scss'; + +const uniqueifyUsers = (users: PrimalNotifUser[]) => { + return users.reduce((acc, u) => { + const found = acc.find(a => a.id === u.id); + return found ? acc : [...acc, u]; + }, []); +} + +const NotificationsSidebar: Component<{ + notifications: SortedNotifications, + getUsers: (notifs: PrimalNotification[], type: NotificationType) => PrimalNotifUser[], +}> = (props) => { + + const intl = useIntl(); + + const follows = () => { + const followNotifs = props.notifications[NotificationType.NEW_USER_FOLLOWED_YOU] || []; + const unffolowNotifs = props.notifications[NotificationType.USER_UNFOLLOWED_YOU] || []; + + const followers = props.getUsers(followNotifs, NotificationType.USER_UNFOLLOWED_YOU); + const lost = props.getUsers(unffolowNotifs, NotificationType.USER_UNFOLLOWED_YOU); + + return [uniqueifyUsers(followers).length, uniqueifyUsers(lost).length]; + + + }; + + const mentions = () => { + const myMentionNotifs = props.notifications[NotificationType.YOU_WERE_MENTIONED_IN_POST] || []; + const myPostMentionNotifs = props.notifications[NotificationType.YOUR_POST_WAS_MENTIONED_IN_POST] || []; + + return [myMentionNotifs.length, myPostMentionNotifs.length]; + }; + + const zaps = () => { + const zapNotifs = props.notifications[NotificationType.YOUR_POST_WAS_ZAPPED] || []; + + const sats = zapNotifs.reduce((acc, n) => { + return n.satszapped ? acc + n.satszapped : acc; + }, 0); + + return [zapNotifs.length, sats]; + }; + + const activity = () => { + const replyNotifs = props.notifications[NotificationType.YOUR_POST_WAS_REPLIED_TO] || []; + const repostNotifs = props.notifications[NotificationType.YOUR_POST_WAS_REPOSTED] || []; + const likeNotifs = props.notifications[NotificationType.YOUR_POST_WAS_LIKED] || []; + + return [replyNotifs.length, repostNotifs.length, likeNotifs.length]; + }; + + const otherNotifications = () => { + const zapedMentionPostNotifs = props.notifications[NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_ZAPPED] || []; + const replyMentionPostNotifs = props.notifications[NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_LIKED] || []; + const repostMentionPostNotifs = props.notifications[NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_REPOSTED] || []; + const likeMentionPostNotifs = props.notifications[NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_REPLIED_TO] || []; + + const zapedPostMentionPostNotifs = props.notifications[NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_ZAPPED] || []; + const replyPostMentionPostNotifs = props.notifications[NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_LIKED] || []; + const repostPostMentionPostNotifs = props.notifications[NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPOSTED] || []; + const likePostMentionPostNotifs = props.notifications[NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPLIED_TO] || []; + + return [ + zapedMentionPostNotifs.length, + replyMentionPostNotifs.length, + repostMentionPostNotifs.length, + likeMentionPostNotifs.length, + + zapedPostMentionPostNotifs.length, + replyPostMentionPostNotifs.length, + repostPostMentionPostNotifs.length, + likePostMentionPostNotifs.length, + ]; + }; + + const otherNotifLabels = [ + { + id: 'notifications.sidebar.mentionsPostZap', + defaultMessage: `{number, plural, + =0 {} + one {mention was zapped} + other {mentions were zapped}}`, + description: 'Sidebar "posts you were mentioned in were zapped" stats description on the notification page', + }, + { + id: 'notifications.sidebar.mentionsPostLike', + defaultMessage: `{number, plural, + =0 {} + one {mention was liked} + other {mentions were liked}}`, + description: 'Sidebar "posts you were mentioned in were liked" stats description on the notification page', + }, + { + id: 'notifications.sidebar.mentionsPostReposted', + defaultMessage: `{number, plural, + =0 {} + one {mention was reposted} + other {mentions were reposted}}`, + description: 'Sidebar "posts you were mentioned in were reposted" stats description on the notification page', + }, + { + id: 'notifications.sidebar.mentionsPostReplied', + defaultMessage: `{number, plural, + =0 {} + one {mention was replied to} + other {mentions were replied to}}`, + description: 'Sidebar "posts you were mentioned in were replied to" stats description on the notification page', + }, + + { + id: 'notifications.sidebar.postMentionsPostZaped', + defaultMessage: `{number, plural, + =0 {} + one {post mention was zapped} + other {post mentions were zapped}}`, + description: 'Sidebar "posts your posts were mentioned in were zapped" stats description on the notification page', + }, + { + id: 'notifications.sidebar.postMentionsPostLike', + defaultMessage: `{number, plural, + =0 {} + one {post mention was liked} + other {post mentions were liked}}`, + description: 'Sidebar "posts your posts were mentioned in were liked" stats description on the notification page', + }, + { + id: 'notifications.sidebar.postMentionsPostReposted', + defaultMessage: `{number, plural, + =0 {} + one {post mention was reposted} + other {post mentions were reposted}}`, + description: 'Sidebar "posts your posts were mentioned in were reposted" stats description on the notification page', + }, + { + id: 'notifications.sidebar.postMentionsPostReposted', + defaultMessage: `{number, plural, + =0 {} + one {post mention was replied to} + other {post mentions were replied to}}`, + description: 'Sidebar "posts your posts were mentioned in were replied to" stats description on the notification page', + }, + ]; + + const nothingNew = () => { + return mentions()[0] + mentions()[1] + + follows()[0] + follows()[1] + + zaps()[0] + + activity()[0] + activity()[1] + activity()[2] === 0; + } + + return ( + <> +
+ {intl.formatMessage(t.heading)} +
+ + +
+ {intl.formatMessage(t.empty)} +
+
+ + 0}> +
+
+
+
+
+
+ {intl.formatMessage(t.followers)} +
+
+
+ 0}> +
{truncateNumber(follows()[0])}
+ {intl.formatMessage( + t.gainedFollowers, + { + number: follows()[0], + }, + )} +
+
+
+ 0}> +
{truncateNumber(follows()[1])}
+ {intl.formatMessage( + t.lostFollowers, + { + number: follows()[1], + }, + )} +
+
+
+
+
+
+ + 0}> +
+
+
+
+
+
+ {intl.formatMessage(t.zaps)} +
+
+
+ 0}> +
{truncateNumber(zaps()[0])}
+ {intl.formatMessage( + t.zapNumber, + { + number: zaps()[0], + }, + )} +
+
+
+ 0}> +
{truncateNumber(zaps()[1])}
+ {intl.formatMessage( + t.statsNumber, + { + number: zaps()[1], + }, + )} +
+
+
+
+
+
+ + 0}> +
+
+
+
+
+
+ {intl.formatMessage(t.activities)} +
+
+
+ 0}> +
{truncateNumber(activity()[0])}
+ {intl.formatMessage( + t.replies, + { + number: activity()[0], + } + )} +
+
+
+ 0}> +
{truncateNumber(activity()[1])}
+ {intl.formatMessage( + t.reposts, + { + number: activity()[1], + }, + )} +
+
+
+ 0}> +
{truncateNumber(activity()[2])}
+ {intl.formatMessage( + t.likes, + { + number: activity()[2], + } + )} +
+
+
+
+
+
+ + 0}> +
+
+
+
+
+
+ {intl.formatMessage(t.mentions)} +
+
+
+ 0}> +
{truncateNumber(mentions()[0])}
+ {intl.formatMessage( + t.mentionsYou, + { + number: mentions()[0], + } + )} +
+
+
+ 0}> +
{truncateNumber(mentions()[1])}
+ {intl.formatMessage( + t.mentionsYourPost, + { + number: mentions()[1], + } + )} +
+
+
+
+
+
+ + 0} + > +
+
+
+
+
+
+ {intl.formatMessage(t.other)} +
+
+ + {(stat, index) => ( + 0}> +
+
+ {truncateNumber(stat)} +
+ {intl.formatMessage( + otherNotifLabels[index()], + { + number: stat, + } + )} +
+
+ )} +
+
+
+
+
+ + ) +} + +export default NotificationsSidebar; diff --git a/src/components/PageNav/PageNav.module.scss b/src/components/PageNav/PageNav.module.scss new file mode 100644 index 0000000..d22cb70 --- /dev/null +++ b/src/components/PageNav/PageNav.module.scss @@ -0,0 +1,34 @@ +@mixin navButton { + display: inline-block; + border: none; + box-shadow: none; + background-color: unset; + margin: 0px; + padding: 0px; + width: 12px; + height: 20px; +} + +.backIcon { + @include navButton; + margin-right: 28px; + + background-color: var(--text-primary); + -webkit-mask: url(../../assets/icons/back.svg) no-repeat center; + mask: url(../../assets/icons/back.svg) no-repeat center; +} + +.forwardIcon { + @include navButton; + + background-color: var(--text-primary); + -webkit-mask: url(../../assets/icons/forward.svg) no-repeat center; + mask: url(../../assets/icons/forward.svg) no-repeat center; +} + + +@media only screen and (max-width: 1300px) { + .backIcon { + margin-right: 8px; + } +} diff --git a/src/components/PageNav/PageNav.tsx b/src/components/PageNav/PageNav.tsx new file mode 100644 index 0000000..89a87c2 --- /dev/null +++ b/src/components/PageNav/PageNav.tsx @@ -0,0 +1,25 @@ +import type { Component } from 'solid-js'; + +import styles from './PageNav.module.scss'; + +const PageNav: Component = () => { + + const onBack = () => { + window.history.back(); + } + + const onNext = () => { + window.history.forward(); + } + + return ( + <> + + + + ) +} + +export default PageNav; diff --git a/src/components/PageTitle/PageTitle.tsx b/src/components/PageTitle/PageTitle.tsx new file mode 100644 index 0000000..9d4d518 --- /dev/null +++ b/src/components/PageTitle/PageTitle.tsx @@ -0,0 +1,28 @@ +import { Component, createEffect, onCleanup } from 'solid-js'; + +const titleTag = document.querySelector('title'); +let origTitle = titleTag?.innerText || ''; + +const PageTitle: Component<{ title: string }> = (props) => { + + createEffect(() => { + + if (titleTag) { + titleTag.innerText = props.title; + } + }); + + onCleanup(() => { + if (titleTag) { + titleTag.innerText = origTitle; + } + + }); + + return ( + <> + + ) +} + +export default PageTitle; diff --git a/src/components/Paginator/Paginator.module.scss b/src/components/Paginator/Paginator.module.scss new file mode 100644 index 0000000..092e2a1 --- /dev/null +++ b/src/components/Paginator/Paginator.module.scss @@ -0,0 +1,17 @@ +.paginator { + color: var(--text-tertiary-2); + position: absolute; + bottom: 0px; + width: 100%; + height: 1280px; + pointer-events: none; +} + +.smallPaginator { + color: var(--text-tertiary-2); + position: relative; + top: 0px; + width: 100%; + height: 280px; + pointer-events: none; +} diff --git a/src/components/Paginator/Paginator.tsx b/src/components/Paginator/Paginator.tsx new file mode 100644 index 0000000..52998e0 --- /dev/null +++ b/src/components/Paginator/Paginator.tsx @@ -0,0 +1,34 @@ +import { onCleanup, onMount } from "solid-js"; +import styles from "./Paginator.module.scss"; + +export default function Paginator(props: { + loadNextPage: (() => void) | undefined, + isSmall?: boolean, +}) { + let observer: IntersectionObserver | undefined; + + onMount(() => { + observer = new IntersectionObserver(entries => { + entries.forEach((entry) => { + if (entry.isIntersecting) { + props.loadNextPage && props.loadNextPage(); + } + }); + }); + + const pag = document.getElementById('pagination_trigger'); + + pag && observer?.observe(pag); + }); + + onCleanup(() => { + const pag = document.getElementById('pagination_trigger'); + + pag && observer?.unobserve(pag); + }); + + return ( +
+
+ ) +} diff --git a/src/components/ParsedNote/ParsedNote.module.scss b/src/components/ParsedNote/ParsedNote.module.scss new file mode 100644 index 0000000..e86e200 --- /dev/null +++ b/src/components/ParsedNote/ParsedNote.module.scss @@ -0,0 +1,90 @@ +.mentionedUser { + color: var(--accent-1); + text-decoration: none; +} + +.mentionedNote { + border: solid 1px var(--subtile-devider); + border-radius: 4px; + margin-block: 6px; + padding: 18px; + display: grid; + grid-template-columns: 1fr; + grid-template-rows: 32px 1fr; + grid-row-gap: 8px; + text-decoration: none; + color: unset; + line-height: 20px; + + &:hover { + text-decoration: none !important; + } + + .mentionedNoteHeader { + display: flex; + justify-content: flex-start; + align-items: center; + color: var(--text-tertiary-2); + + .postInfo { + display: flex; + justify-content: flex-start; + margin-left: 11px; + + .userInfo { + font-size: 14px; + line-height: 16px; + font-weight: 400; + display: flex; + align-items: center; + width: auto; + .userName { + max-width: 150px; + text-overflow: ellipsis; + white-space: nowrap; + overflow: hidden; + } + } + + .time{ + margin: 0px 2px; + min-width: 40px; + font-size: 14px; + line-height: 16px; + font-weight: 400; + &::before { + content: "|"; + padding: 0px 2px; + } + } + + .contextMenu { + min-width: 5px; + display: inline-block; + text-align: center; + font-weight: bold; + } + } + } +} + +.verifiedIcon { + width: 13px; + height: 12px; + display: inline-block; + margin: 0px 2px; + background-color: var(--text-tertiary-2); + -webkit-mask: url(../../assets/icons/verified.svg) no-repeat center; + mask: url(../../assets/icons/verified.svg) no-repeat center; +} + +.bordered { + border:solid 1px var(--subtile-devider); + border-radius: 8px; + margin-block: 8px; + overflow: hidden; +} + +.error { + color: var(--brand-1); +} diff --git a/src/components/ParsedNote/ParsedNote.tsx b/src/components/ParsedNote/ParsedNote.tsx new file mode 100644 index 0000000..e9a3f43 --- /dev/null +++ b/src/components/ParsedNote/ParsedNote.tsx @@ -0,0 +1,231 @@ +import { A } from '@solidjs/router'; +import { hexToNpub } from '../../lib/keys'; +import { linkPreviews, parseNote1 } from '../../lib/notes'; +import { truncateNpub, userName } from '../../stores/profile'; +import EmbeddedNote from '../EmbeddedNote/EmbeddedNote'; +import { + Component, createEffect, createSignal, +} from 'solid-js'; +import { + PrimalNote, +} from '../../types/primal'; + +import styles from './ParsedNote.module.scss'; +import { nip19 } from 'nostr-tools'; +import LinkPreview from '../LinkPreview/LinkPreview'; +import MentionedUserLink from '../Note/MentionedUserLink/MentionedUserLink'; + + +export const parseNoteLinks = (text: string, note: PrimalNote, highlightOnly = false) => { + + const regex = /\bnostr:((note|nevent)1\w+)\b|#\[(\d+)\]/g; + + return text.replace(regex, (url) => { + const [_, id] = url.split(':'); + + if (!id) { + return url; + } + + try { + const eventId = nip19.decode(id).data as string | nip19.EventPointer; + const hex = typeof eventId === 'string' ? eventId : eventId.id; + const noteId = nip19.noteEncode(hex); + + const path = `/thread/${noteId}`; + + const ment = note.mentionedNotes && note.mentionedNotes[hex]; + + const link = highlightOnly ? + {url} : + ment ? +
+ +
: + {url}; + + // @ts-ignore + return link.outerHTML || url; + } catch (e) { + return `${url}`; + } + + }); + +}; + +export const parseNpubLinks = (text: string, note: PrimalNote, highlightOnly = false) => { + + const regex = /\bnostr:((npub|nprofile)1\w+)\b|#\[(\d+)\]/g; + + return text.replace(regex, (url) => { + const [_, id] = url.split(':'); + + if (!id) { + return url; + } + + try { + const profileId = nip19.decode(id).data as string | nip19.ProfilePointer; + + const hex = typeof profileId === 'string' ? profileId : profileId.pubkey; + const npub = hexToNpub(hex); + + const path = `/profile/${npub}`; + + const user = note.mentionedUsers && note.mentionedUsers[hex]; + + let link = highlightOnly ? + @{truncateNpub(npub)} : + @{truncateNpub(npub)}; + + if (user) { + link = highlightOnly ? + @{userName(user)} : + MentionedUserLink({ user }); + } + + // @ts-ignore + return link.outerHTML || url; + } catch (e) { + return `${url}`; + } + }); + +}; + +const ParsedNote: Component<{ note: PrimalNote, ignoreMentionedNotes?: boolean}> = (props) => { + + const parsedContent = (text: string) => { + const regex = /\#\[([0-9]*)\]/g; + let parsed = text; + + let refs = []; + let match; + + while((match = regex.exec(text)) !== null) { + refs.push(match[1]); + } + + if (refs.length > 0) { + for(let i =0; i < refs.length; i++) { + let r = parseInt(refs[i]); + + const tag = props.note.post.tags[r]; + + if (tag === undefined || tag.length === 0) continue; + + if ( + tag[0] === 'e' && + props.note.mentionedNotes && + props.note.mentionedNotes[tag[1]] + ) { + const embeded = ( +
+ +
+ ); + + + // @ts-ignore + parsed = parsed.replace(`#[${r}]`, embeded.outerHTML); + } + + if (tag[0] === 'p' && props.note.mentionedUsers && props.note.mentionedUsers[tag[1]]) { + const user = props.note.mentionedUsers[tag[1]]; + + const link = MentionedUserLink({ user }); + + // @ts-ignore + parsed = parsed.replace(`#[${r}]`, link.outerHTML); + } + } + } + + return parsed; + + }; + + const highlightHashtags = (text: string) => { + const regex = /(?:\s|^)#[^\s!@#$%^&*(),.?":{}|<>]+/ig; + + return text.replace(regex, (token) => { + const [space, term] = token.split('#'); + const embeded = ( + + {space} + #{term} + + ); + + // @ts-ignore + return embeded.outerHTML; + }); + } + + const replaceLinkPreviews = (text: string, previews: Record) => { + let parsed = text; + + const regex = /__LINK__.*?__LINK__/ig; + + parsed = parsed.replace(regex, (link) => { + const url = link.split('__LINK__')[1]; + + const preview = previews[url]; + + // No preview? That can only mean that we are still waiting. + if (!preview) { + return link; + } + + if (preview.noPreview) { + return `${url}`; + } + + const linkElement = (
); + + // @ts-ignore + return linkElement.outerHTML; + }); + + return parsed; + } + + const content = () => { + return parseNoteLinks( + parseNpubLinks( + parsedContent( + highlightHashtags( + parseNote1(props.note.post.content) + ), + ), + props.note, + ), + props.note, + ); + }; + + const [displayedContent, setDisplayedContent] = createSignal(content()); + + createEffect(() => { + const newContent = replaceLinkPreviews(displayedContent(), { ...linkPreviews }); + + setDisplayedContent(() => newContent); + }); + + + return ( +
+
+ ); +}; + +export default ParsedNote; diff --git a/src/components/PeopleList/PeopleList.module.scss b/src/components/PeopleList/PeopleList.module.scss new file mode 100644 index 0000000..fb90885 --- /dev/null +++ b/src/components/PeopleList/PeopleList.module.scss @@ -0,0 +1,179 @@ +@mixin heading { + position: -webkit-sticky; + position: sticky; + top: 0px; + width: 100%; + height: 44px; + // background-color: var(--background-site); + background: var(--fade-gradient-vertical); + z-index: 5; + padding-bottom: 22px; + display:flex; + flex-direction: row; + align-items: center; + justify-content: flex-start; + font-size: 18px; + font-weight: 800; + line-height: 20px; + color: var(--text-secondary-2); + text-transform: uppercase; + >div{ + height: 22px; + >span { + color: var(--text-tertiary-2); + text-transform: lowercase; + margin-left: 6px; + } + } +} + +.trendingSection { + // position: -webkit-sticky; + // position: sticky; + // top: 0px; + padding-top: 44px; + margin-bottom: 228px +} + +.stickyWrapper { + height: 100%; + position: fixed; + height: 100%; + overflow-y: scroll; + + &::-webkit-scrollbar{ + display: none; + } +} + +.heading { + @include heading(); +} + +.peopleList { + margin-bottom: 33px; + display: grid; + grid-template-columns: 52px 156px 64px; + grid-template-rows: 1fr; + grid-template-areas: "avatar content follow"; + grid-column-gap: 14px; + text-decoration: none; + + .avatar { + grid-area: avatar; + display: grid; + justify-items: center; + height: 52px; + + .avatarImg { + width: 52px; + height: 52px; + border-radius: 50%; + } + } + + .content { + grid-area: content; + display: flex; + flex-direction: column; + justify-content: flex-start; + + .name { + color: var(--text-primary); + font-weight: 400; + font-size: 12px; + line-height: 12px; + text-overflow: ellipsis; + white-space: nowrap; + overflow: hidden; + } + + .verification { + font-size: 12px; + line-height: 16px; + font-weight: 400; + color: var(--text-tertiary-2); + text-overflow: ellipsis; + white-space: nowrap; + overflow: hidden; + } + + .verifiedName { + font-size: 12px; + line-height: 16px; + font-weight: 700; + color: var(--text-tertiary-2); + text-overflow: ellipsis; + white-space: nowrap; + overflow: hidden; + } + + .npub { + font-size: 12px; + line-height: 16px; + font-weight: 300; + color: var(--text-tertiary-2); + text-overflow: ellipsis; + white-space: nowrap; + overflow: hidden; + } + + &:hover { + .message, .name, .time { + color: var(--text-primary); + text-decoration: underline; + } + + cursor: pointer; + } + } + + @mixin followButton { + grid-area: follow; + align-items: center; + display: flex; + align-items: center; + button { + width: 64px; + height: 40px; + background: var(--brand-gradient); + border-radius: 6px; + padding: 0px; + font-size: 12px; + line-height: 16px; + margin: 0px; + } + + } + + .follow { + @include followButton; + } + + .unfollow { + @include followButton; + button { + background-color: var(--background-card); + background: linear-gradient(var(--background-card), var(--background-card)) padding-box, + var(--brand-gradient) border-box; + border: 1px solid transparent; + } + } + + &:hover { + .name { + text-decoration: underline; + color: var(--text-primary); + } + } +} + +.verifiedIcon { + width:13px; + height: 12px; + display: inline-block; + margin: 0px 2px; + background-color: var(--text-tertiary-2); + -webkit-mask: url(../../assets/icons/verified.svg) no-repeat center; + mask: url(../../assets/icons/verified.svg) no-repeat center; +} diff --git a/src/components/PeopleList/PeopleList.tsx b/src/components/PeopleList/PeopleList.tsx new file mode 100644 index 0000000..cfcbba6 --- /dev/null +++ b/src/components/PeopleList/PeopleList.tsx @@ -0,0 +1,56 @@ +import { A } from '@solidjs/router'; +import { Component, For, Show } from 'solid-js'; +import { authorName, nip05Verification, truncateNpub } from '../../stores/profile'; +import { PrimalUser } from '../../types/primal'; +import Avatar from '../Avatar/Avatar'; +import FollowButton from '../FollowButton/FollowButton'; + +import styles from './PeopleList.module.scss'; + + +const PeopleList: Component<{ people: PrimalUser[], label: string}> = (props) => { + const people = () => props.people; + + return ( + + ); +} + +export default PeopleList; diff --git a/src/components/PostButton/PostButton.module.scss b/src/components/PostButton/PostButton.module.scss new file mode 100644 index 0000000..09d3d42 --- /dev/null +++ b/src/components/PostButton/PostButton.module.scss @@ -0,0 +1,19 @@ +.postButton { + width: 36px; + height: 36px; + padding: 0px; + border-radius: 0px 6px 6px 0px; + margin: 0px; + background: var(--brand-gradient); +} + +.postIcon { + display: inline-block; + width: 24px; + height: 24px; + vertical-align: middle; + + background-color: white; + -webkit-mask: url(../../assets/icons/post.svg) no-repeat center; + mask: url(../../assets/icons/post.svg) no-repeat center; +} diff --git a/src/components/PostButton/PostButton.tsx b/src/components/PostButton/PostButton.tsx new file mode 100644 index 0000000..5d8716f --- /dev/null +++ b/src/components/PostButton/PostButton.tsx @@ -0,0 +1,15 @@ +import styles from "./PostButton.module.scss"; + +export default function PostButton() { + + const showPostForm = () => {}; + + return ( + + ) +} diff --git a/src/components/ProfileSidebar/ProfileSidebar.module.scss b/src/components/ProfileSidebar/ProfileSidebar.module.scss new file mode 100644 index 0000000..498bfdb --- /dev/null +++ b/src/components/ProfileSidebar/ProfileSidebar.module.scss @@ -0,0 +1,38 @@ +@mixin heading { + position: -webkit-sticky; + position: sticky; + top: 0px; + width: 100%; + height: 44px; + // background-color: var(--background-site); + background: var(--fade-gradient-vertical); + z-index: 5; + padding-bottom: 22px; + display:flex; + flex-direction: row; + align-items: center; + justify-content: flex-start; + font-size: 18px; + font-weight: 800; + line-height: 22px; + color: var(--text-secondary-2); + text-transform: uppercase; + >div{ + display: flex; + height: 22px; + >span { + color: var(--text-tertiary-2); + text-transform: lowercase; + margin-left: 6px; + } + } +} + +.headingTrending { + @include heading(); +} + +.noNotes { + color: var(--text-tertiary-2); + text-transform: lowercase; +} diff --git a/src/components/ProfileSidebar/ProfileSidebar.tsx b/src/components/ProfileSidebar/ProfileSidebar.tsx new file mode 100644 index 0000000..94fecbb --- /dev/null +++ b/src/components/ProfileSidebar/ProfileSidebar.tsx @@ -0,0 +1,47 @@ +import { Component, For, Show } from 'solid-js'; +import { + PrimalNote, + PrimalUser +} from '../../types/primal'; + +import styles from './ProfileSidebar.module.scss'; +import SmallNote from '../SmallNote/SmallNote'; +import { useIntl } from '@cookbook/solid-intl'; +import { userName } from '../../stores/profile'; +import { profile as t } from '../../translations'; + + +const ProfileSidebar: Component<{ notes: PrimalNote[] | undefined, profile: PrimalUser | undefined }> = (props) => { + + const intl = useIntl(); + + return ( + +
+
+ {intl.formatMessage(t.sidebarCaption)} +
+
+ + 0} + fallback={ +
+ {intl.formatMessage( + t.sidebarNoNotes, + { + name: userName(props.profile), + }, + )} +
+ } + > + + {(note) => } + +
+
+ ); +} + +export default ProfileSidebar; diff --git a/src/components/ProfileWidget/ProfileWidget.module.scss b/src/components/ProfileWidget/ProfileWidget.module.scss new file mode 100644 index 0000000..309e615 --- /dev/null +++ b/src/components/ProfileWidget/ProfileWidget.module.scss @@ -0,0 +1,113 @@ +.userProfile { + display: grid; + width: 146px; + height: 46px; + grid-template-columns: 42px 1fr; + grid-column-gap: 5px; + position: relative; + align-items: center; + text-decoration: none; + + .background { + position: absolute; + top: 0; + left: 0; + width: 146px; + height: 46px; + background-color: var(--background-input); + background: linear-gradient(var(--background-input), var(--background-input)) padding-box, + var(--brand-gradient) border-box; + border-radius: 21px; + border: 1px solid transparent; + opacity: 0.5; + transition: opacity 0.4s; + + } + + &:hover { + .background { + opacity: 1; + transition: opacity 0.4s; + } + } + +} +.avatar { + z-index: 1; + padding: 2px; +} + +.userInfo { + display: flex; + flex-direction: column; + align-items: flex-start; + justify-content: center; + z-index: 1; + padding: 2px; +} + +.userName { + font-size: 12px; + line-height: 16px; + font-weight: 700; + color: var(--text-primary); + width: 90px; + text-overflow: ellipsis; + white-space: nowrap; + overflow: hidden; +} + +.userVerification { + font-size: 12px; + line-height: 16px; + font-weight: 400; + color: var(--text-tertiary); + width: 92px; + text-overflow: ellipsis; + white-space: nowrap; + overflow: hidden; +} + +.contextMenu { + display: flex; + align-items: center; +} + +.contextIcon { + display: inline-block; + width: 10px; + height: 2px; + vertical-align: middle; + + background-color: var(--text-secondary); + -webkit-mask: url(../../assets/icons/context.svg) no-repeat center; + mask: url(../../assets/icons/context.svg) no-repeat center; +} + +@media only screen and (max-width: 1300px) { + .userProfile { + display: grid; + width: 48px; + grid-template-columns: 48px; + grid-column-gap: 0px; + + .background { + width: 46px; + height: 46px; + } + } + + .userInfo { + display: none; + } + + .contextMenu { + display: none; + } +} + +@media only screen and (max-width: 720px) { + .userProfile { + display: none; + } +} diff --git a/src/components/ProfileWidget/ProfileWidget.tsx b/src/components/ProfileWidget/ProfileWidget.tsx new file mode 100644 index 0000000..2e2ebda --- /dev/null +++ b/src/components/ProfileWidget/ProfileWidget.tsx @@ -0,0 +1,40 @@ +import { Component, Show } from 'solid-js'; +import { A } from '@solidjs/router'; +import Avatar from '../Avatar/Avatar'; +import { useAccountContext } from '../../contexts/AccountContext'; +import { trimVerification } from '../../lib/profile'; +import { hexToNpub } from '../../lib/keys'; + +import styles from './ProfileWidget.module.scss'; + +const ProfileWidget: Component = () => { + + const account = useAccountContext() + + const activeUser = () => account?.activeUser; + + return ( + + ); +} + +export default ProfileWidget; diff --git a/src/components/ReplyToNote/ReplyToNote.module.scss b/src/components/ReplyToNote/ReplyToNote.module.scss new file mode 100644 index 0000000..714a3c6 --- /dev/null +++ b/src/components/ReplyToNote/ReplyToNote.module.scss @@ -0,0 +1,175 @@ +.newNoteBorder { + width: 100%; + min-height: 120px; + padding: 1px; + background: var(--brand-gradient); + border-radius: 6px; + display: block; + position: relative; + margin-block: 5px; +} + +.newNote { + width: 100%; + height: 100%; + min-height: 120px; + font-size: 18px; + line-height: 20px; + margin: 0px; + border-radius: 6px; + border: none; + color: var(--text-tertiary); + background-color: var(--background-site); + display: grid; + grid-template-columns: 92px 1fr; + + .leftSide { + padding: 16px 21px; + } +} + +.controls { + display: flex; + justify-content: flex-end; + align-items: center; + margin: 0px 12px 12px 0px; + >button { + width: 80px; + height: 28px; + margin: 0px 0px 0px 8px; + } +} + +.primaryButton { + border: none; + border-radius: 6px; + margin: 0px 8px; + padding: 0px; + font-size: 14px; + line-height: 20px; + font-weight: 700; + background: var(--brand-gradient-vertical); + color: var(--text-primary); + >span { + opacity: 0.75; + } +} + + +.secondaryButton { + border: none; + border-radius: 6px; + margin: 0px 15px 0px 0px; + padding: 1px; + font-size: 14px; + line-height: 20px; + font-weight: 700; + background: var(--brand-gradient-vertical); + color: var(--text-tertiary-2); + >div { + width: 100%; + height: 100%; + vertical-align: middle; + border-radius: 6px; + background-color: var(--background-card); + display: flex; + flex-direction: column; + align-items: center; + justify-content: center; + } +} + +.border { + height: 36px; + padding: 1px; + background: var(--brand-gradient); + border-radius: 6px; + + .input { + height: 34px; + width: 100%; + font-size: 18px; + line-height: 20px; + margin: 0px; + border-radius: 6px; + border: none; + color: var(--text-tertiary); + background-color: var(--background-site); + display: flex; + align-items: center; + padding-inline: 12px; + + .userName { + max-width: 408px; + text-overflow: ellipsis; + white-space: nowrap; + word-wrap: break-word; + overflow: hidden; + margin-left: 6px; + } + } +} + +.replyBox { + margin: 4px 0px; + // padding: 30px 22px; + background-color: var(--background-card); + border: none; + outline: none; + display: grid; + grid-template-columns: 72px 1fr; +} + +.leftSideClosed { + // margin-top: -13px; + // margin-left: -1px; + padding-top: 3px; + padding-left: 2px; +} + +.rightSideClosed { + padding-top: 10px; +} + +@media only screen and (max-width: 720px) { + .border { + height: 36px; + padding: 1px; + background: var(--brand-gradient); + border-radius: 6px; + margin-left: 10px; + + .input { + height: 34px; + font-size: 18px; + line-height: 20px; + margin: 0px; + border-radius: 6px; + border: none; + color: var(--text-tertiary); + background-color: var(--background-site); + display: flex; + align-items: center; + padding-left: 12px; + } + } + + .replyBox { + margin-top: 4px; + padding: 30px 12px; + width: calc(100vw - (100vw - 100%)); + background-color: var(--background-card); + } +} + + +.searchSuggestions { + width: 300px; + background-color: var(--background-site); + border: 1px solid var(--text-tertiary-2); + // box-shadow: 0px 4px 12px rgba(0, 0, 0, 0.8); + border-radius: 4px; + + position: absolute; + z-index: 50; +} diff --git a/src/components/ReplyToNote/ReplyToNote.tsx b/src/components/ReplyToNote/ReplyToNote.tsx new file mode 100644 index 0000000..9d794d3 --- /dev/null +++ b/src/components/ReplyToNote/ReplyToNote.tsx @@ -0,0 +1,136 @@ +import { useIntl } from "@cookbook/solid-intl"; +import { Component, createEffect, createSignal, Show } from "solid-js"; +import { useAccountContext } from "../../contexts/AccountContext"; +import { userName } from "../../stores/profile"; +import { actions as t } from "../../translations"; +import { PrimalNote } from "../../types/primal"; +import Avatar from "../Avatar/Avatar"; +import EditBox from "../NewNote/EditBox/EditBox"; +import styles from "./ReplyToNote.module.scss"; + +type AutoSizedTextArea = HTMLTextAreaElement & { _baseScrollHeight: number }; + +const getScrollHeight = (elm: AutoSizedTextArea) => { + var savedValue = elm.value + elm.value = '' + elm._baseScrollHeight = elm.scrollHeight + elm.value = savedValue +} + +const onExpandableTextareaInput: (event: InputEvent) => void = (event) => { + + const maxHeight = document.documentElement.clientHeight || window.innerHeight || 0; + + const elm = event.target as AutoSizedTextArea ; + + if(elm.nodeName !== 'TEXTAREA' || elm.id !== 'reply_to_note_text_area') { + return; + } + + const minRows = parseInt(elm.getAttribute('data-min-rows') || '0'); + + !elm._baseScrollHeight && getScrollHeight(elm); + + if (elm.scrollHeight >= (maxHeight - 70)) { + return; + } + + elm.rows = minRows + const rows = Math.ceil((elm.scrollHeight - elm._baseScrollHeight) / 20) + elm.rows = minRows + rows +} + +const ReplyToNote: Component<{ note: PrimalNote }> = (props) => { + + const intl = useIntl(); + + const [open, setOpen] = createSignal(false); + + const account = useAccountContext(); + + const activeUser = () => account?.activeUser; + + const openReplyBox = () => { + setOpen(true); + }; + + const closeReplyToNote = () => { + setOpen(false); + }; + + createEffect(() => { + if (open()) { + setTimeout(() => { + const newNoteTextArea = document.getElementById('reply_new_note_text_area') as HTMLTextAreaElement | undefined; + + if (!newNoteTextArea) { + return; + } + newNoteTextArea?.focus(); + }, 100); + } + else { + const newNoteTextArea = document.getElementById('reply_new_note_text_area') as HTMLTextAreaElement | undefined; + + if (!newNoteTextArea) { + return; + } + + newNoteTextArea.value = ''; + } + }); + + return ( + +
+ +
+
+
+
+ + {intl.formatMessage( + t.noteReply, + { + name: userName(props.note.user), + }, + )} + +
+
+
+ + } + > +
+
+
+ +
+
+ +
+
+
+
+ ) +} + +export default ReplyToNote; diff --git a/src/components/Search/Search.module.scss b/src/components/Search/Search.module.scss new file mode 100644 index 0000000..ee19a60 --- /dev/null +++ b/src/components/Search/Search.module.scss @@ -0,0 +1,89 @@ +.search { + display: grid; + grid-template-columns: 16px 1fr; + grid-column-gap: 15px; + border-radius: 22px; + background-color: var(--background-input); + width: 300px; + padding-right: 22px; + + &:hover { + input { + color: var(--text-primary); + } + + .searchIcon { + background-color: var(--text-primary); + } + } + + input { + height: 36px; + font-weight: 700; + font-size: 16px; + line-height: 20px; + padding: 0px 12px; + margin-bottom: 0px; + border: none; + background-color: var(--background-input); + color: var(--text-tertiary); + + &:focus { + border: none; + box-shadow: none; + } + + + &::placeholder { + color: var(--text-tertiary); + } + } + + .searchIcon { + display: inline-block; + width: 16px; + height: 16px; + margin-left: 16px; + margin-top: 10px; + + background-color: var(--text-tertiary); + -webkit-mask: url(../../assets/icons/search.svg) 0 0/16px 16px; + mask: url(../../assets/icons/search.svg) 0 0/16px 16px; + } +} + +.searchHolder { + position: relative; + width: 300px; + padding-top: 26px; +} + +.searchSuggestions { + width: 300px; + background-color: var(--background-site); + border: 1px solid var(--text-tertiary-2); + // box-shadow: 0px 4px 12px rgba(0, 0, 0, 0.8); + border-radius: 4px; + + position: absolute; + top: 76px; +} + +.loadingOverlay { + position: absolute; + width: 100%; + height: 100%; + z-index: 10; + background-color: var(--background-site); + opacity: 0.8; +} + +.searchIcon { + display: inline-block; + width: 18px; + height: 18px; + + background-color: var(--text-primary); + -webkit-mask: url(../../assets/icons/search.svg) 0 0/18px 18px; + mask: url(../../assets/icons/search.svg) 0 0/18px 18px; +} diff --git a/src/components/Search/Search.tsx b/src/components/Search/Search.tsx new file mode 100644 index 0000000..96e30bf --- /dev/null +++ b/src/components/Search/Search.tsx @@ -0,0 +1,191 @@ +import { useIntl } from '@cookbook/solid-intl'; +import { useNavigate } from '@solidjs/router'; +import { Component, createEffect, createSignal, For, Show } from 'solid-js'; +import { useSearchContext } from '../../contexts/SearchContext'; +import { nip05Verification, userName } from '../../stores/profile'; +import { PrimalUser } from '../../types/primal'; +import { debounce } from '../../utils'; +import Avatar from '../Avatar/Avatar'; +import Loader from '../Loader/Loader'; +import { useToastContext } from '../Toaster/Toaster'; +import { placeholders, search as t } from '../../translations'; + +import styles from './Search.module.scss'; +import SearchOption from './SearchOption'; + + +const Search: Component<{ + onInputConfirm?: (query: string) => void, + onUserSelect?: (selected: PrimalUser | string) => void, + noLinks?: boolean, + hideDefault?: boolean, + placeholder?: string, +}> = (props) => { + + const toaster = useToastContext(); + const search = useSearchContext(); + const navigate = useNavigate(); + const intl = useIntl(); + + const [query, setQuery] = createSignal(''); + const [isFocused, setIsFocused] = createSignal(false); + + const queryUrl = () => query().replaceAll('#', '%23'); + + let input: HTMLInputElement | undefined; + + const onSearch = (e: SubmitEvent) => { + e.preventDefault(); + + const form = e.target as HTMLFormElement; + + const data = new FormData(form); + + const q = data.get('searchQuery') as string || ''; + + if (q.length > 0) { + if (props.onInputConfirm) { + props.onInputConfirm(q); + } + else { + navigate(`/search/${q.replaceAll('#', '%23')}`); + } + onBlur(); + resetQuery(); + } + else { + toaster?.sendInfo(intl.formatMessage(t.invalid)) + } + return false; + } + + const onInput = (e: InputEvent) => { + setIsFocused(true); + debounce(() => { + // @ts-ignore + const value = e.target?.value; + + if (value.startsWith('npub') || value.startsWith('nprofile')) { + search?.actions.findUserByNupub(value); + return; + } + + setQuery(value || ''); + }, 500); + }; + + const onFocus = (e: FocusEvent) => { + setIsFocused(true); + } + + const onBlur = (e?: FocusEvent) => { + setTimeout(() => { + setIsFocused(false); + }, 200); + } + + const resetQuery = () => { + setQuery(''); + + if (input) { + input.value = ''; + } + }; + + const selectUser = (user: PrimalUser) => { + if (props.onUserSelect) { + props.onUserSelect(user); + } + resetQuery(); + } + + createEffect(() => { + if (!isFocused()) { + return; + } + + if (query().length === 0) { + search?.actions.getRecomendedUsers(); + return; + } + + search?.actions.findUsers(query()); + }); + + return ( +
+
+
+ +
+ + +
+ +
+ +
+
+ + 0} + fallback={ +
} + underline={true} + /> + } + > +
} + underline={true} + onClick={resetQuery} + /> + + + + + {(user) => ( + } + statNumber={search?.scores[user.pubkey]} + statLabel={intl.formatMessage(t.followers)} + onClick={() => selectUser(user)} + /> + )} + + + + + + ) +} + +export default Search; diff --git a/src/components/Search/SearchOption.module.scss b/src/components/Search/SearchOption.module.scss new file mode 100644 index 0000000..b927dd6 --- /dev/null +++ b/src/components/Search/SearchOption.module.scss @@ -0,0 +1,75 @@ + +.userResult { + display: grid; + grid-template-columns: 36px 1fr 60px; + padding: 8px; + margin: 4px; + text-decoration: none; + border-radius: 4px; + cursor: pointer; + + &:hover, &:focus, &.highlight { + background-color: var(--background-input); + } + + .userAvatar { + display: flex; + align-items: center; + justify-content: center; + + .searchIcon { + display: inline-block; + width: 18px; + height: 18px; + + background-color: var(--text-primary); + -webkit-mask: url(../../assets/icons/search.svg) 0 0/18px 18px; + mask: url(../../assets/icons/search.svg) 0 0/18px 18px; + } + } + + .userInfo { + margin-left: 6px; + display: flex; + flex-direction: column; + justify-content: center; + max-width: 172px; + } + + .userName { + font-weight: 700; + font-size: 14px; + line-height: 16px; + color: var(--text-primary); + } + + .verification { + font-weight: 400; + font-size: 14px; + line-height: 16px; + color: var(--text-secondary-2); + } + + .userStats { + display: flex; + flex-direction: column; + align-items: flex-end; + .followerNumber { + font-weight: 700; + font-size: 12px; + line-height: 16px; + color: var(--text-primary) + } + + .followerLabel { + font-weight: 400; + font-size: 12px; + line-height: 14px; + color: var(--text-secondary-2); + } + } +} + +.underline { + border-bottom: 1px solid var(--subtile-devider); +} diff --git a/src/components/Search/SearchOption.tsx b/src/components/Search/SearchOption.tsx new file mode 100644 index 0000000..ef29139 --- /dev/null +++ b/src/components/Search/SearchOption.tsx @@ -0,0 +1,80 @@ +import { A } from '@solidjs/router'; +import { Component, JSXElement, Show } from 'solid-js'; +import { truncateNumber } from '../../lib/notifications'; +import { truncateName, } from '../../stores/profile'; + + +import styles from './SearchOption.module.scss'; + + +const SearchOption: Component<{ + href?: string, + title: string, + description?: string, + icon: JSXElement, + statNumber?: number, + statLabel?: string, + underline?: boolean, + onClick?: (e?: MouseEvent) => void, + highlighted?: boolean, +}> = (props) => { + + const Content: Component<{ children: JSXElement }> = (prp) => { + const klass = () => `${styles.userResult} + ${props.underline ? styles.underline : ''} + ${props.highlighted ? styles.highlight : ''}`; + + return ( + + {prp.children} + + } + > + + {prp.children} + + + ); + }; + + return ( + +
+ {props.icon} +
+
+
+ {props.title} +
+ 0}> +
+ {truncateName(props.description || '')} +
+
+
+ +
+
+ {truncateNumber(props.statNumber || 0)} +
+
+ {props.statLabel} +
+
+
+
+ ); +} + +export default SearchOption; diff --git a/src/components/SearchSidebar/SearchSidebar.tsx b/src/components/SearchSidebar/SearchSidebar.tsx new file mode 100644 index 0000000..4b7caeb --- /dev/null +++ b/src/components/SearchSidebar/SearchSidebar.tsx @@ -0,0 +1,21 @@ +import { Component } from 'solid-js'; +import { PrimalUser } from '../../types/primal'; +import { useIntl } from '@cookbook/solid-intl'; +import { search as t } from '../../translations'; +import PeopleList from '../PeopleList/PeopleList'; + + +const SearchSidebar: Component<{ users: PrimalUser[] }> = (props) => { + + const intl = useIntl(); + + return ( + <> + + + ); +} + +export default SearchSidebar; diff --git a/src/components/SelectBox/SelectBox.scss b/src/components/SelectBox/SelectBox.scss new file mode 100644 index 0000000..f1fdea3 --- /dev/null +++ b/src/components/SelectBox/SelectBox.scss @@ -0,0 +1,240 @@ +.feed_select { + height: 20px; + font-size: 14px; + line-height: 16px; + padding: 0px; + margin: 0px; + width: auto; + border: none; + text-align: right; + background-position: center right !important; + color: var(--text-tertiary-2); + padding-right: 6px; + + .solid-select-container { + position: relative; + } + + .solid-select-control { + margin: 0px 0px 0px 8px; + padding: 0; + cursor: pointer; + pointer-events:painted; + width: calc(100% - 8px); + display: flex; + align-items: center; + + &:after { + content: ''; + display: inline-block; + margin-left: 8px; + width: 18px; + height: 18px; + padding-right: 6px; + + background-color: var(--text-tertiary-2); + -webkit-mask: url(../../assets/icons/feed_picker.svg) no-repeat 0 0/ 14px 14px; + mask: url(../../assets/icons/feed_picker.svg) no-repeat 0 0/ 14px 14px; + } + + &:hover { + color: var(--text-primary); + &:after { + background-color: var(--text-primary); + } + } + } + + .highlighted { + color: var(--text-primary); + &:after { + background-color: var(--text-primary); + } + } + + .solid-select-option { + cursor: pointer; + padding-right: 34px; + } + + .solid-select-option:hover { + background-color: var(--background-card); + } + .solid-select-option[data-focused=true] { + background-color: unset; + + &:hover { + background-color: var(--background-card); + } + } + .solid-select-option[data-disabled=true] { + padding-right: 10px; + color: var(--text-primary); + + &:after { + content: ''; + display: inline-block; + margin-left: 14px; + width: 10px; + height: 10px; + + background-color: var(--text-primary); + -webkit-mask: url(../../assets/icons/check.svg) no-repeat center; + mask: url(../../assets/icons/check.svg) no-repeat center; + } + } + + .solid-select-list { + margin-top: 0; + padding: 8px; + border: solid 1px var(--text-tertiary-2); + border-radius: 4px; + background-color: var(--background-site); + right: 0; + } + + .solid-select-input { + max-height: 0px; + height: 0px; + box-shadow: none; + margin: 0; + padding: 0; + font-size: 16px; + cursor: pointer; + } + + .solid-select-single-value { + caret-color: transparent; + text-overflow: ellipsis; + white-space: nowrap; + overflow: hidden; + width: 100%; + } + +} +.selector { + height: 20px; + font-size: 14px; + line-height: 16px; + padding: 0px; + margin: 0px; + width: 179px; + border: none; + text-align: right; + background-position: center right !important; + color: var(--text-secondary); +} + +.phone_feed_select { + position: relative; + height: 20px; + font-size: 18px; + line-height: 18px; + font-weight: 700; + padding: 0px; + margin: 0px; + border: none; + text-align: right; + background-position: center right !important; + color: var(--text-primary); + padding-right: 6px; + width: 100%; + + .solid-select-control { + margin: 0px 0px 0px 8px; + padding: 0; + cursor: pointer; + pointer-events:painted; + width: 100%; + display: flex; + align-items: center; + + &:after { + content: ''; + display: inline-block; + margin-left: 16px; + width: 18px; + height: 18px; + + background-color: var(--text-primary); + -webkit-mask: url(../../assets/icons/feed_picker.svg) no-repeat 0 0/ 14px 14px; + mask: url(../../assets/icons/feed_picker.svg) no-repeat 0 0/ 14px 14px; + } + + &:hover { + color: var(--text-primary); + &:after { + background-color: var(--text-primary); + } + } + } + + .highlighted { + color: var(--text-primary); + &:after { + background-color: var(--text-primary); + } + } + + .solid-select-option { + cursor: pointer; + padding-right: 34px; + color: var(--text-tertiary-2); + padding-block: 24px; + } + + .solid-select-option:hover { + background-color: var(--background-card); + } + .solid-select-option[data-focused=true] { + background-color: unset; + + &:hover { + background-color: var(--background-card); + } + } + .solid-select-option[data-disabled=true] { + padding-right: 10px; + color: var(--text-primary); + + &:after { + content: ''; + display: inline-block; + margin-left: 14px; + width: 10px; + height: 10px; + + background-color: var(--text-primary); + -webkit-mask: url(../../assets/icons/check.svg) no-repeat center; + mask: url(../../assets/icons/check.svg) no-repeat center; + } + } + + .solid-select-list { + margin-top: 0; + padding: 8px; + border: solid 1px var(--text-tertiary-2); + border-radius: 4px; + background-color: var(--background-site); + right: 0; + } + + .solid-select-input { + max-height: 0px; + height: 0px; + box-shadow: none; + margin: 0; + padding: 0; + font-size: 16px; + cursor: pointer; + } + + .solid-select-single-value { + caret-color: transparent; + text-overflow: ellipsis; + white-space: nowrap; + overflow: hidden; + display: flex; + max-width: 100%; + } +} diff --git a/src/components/SelectBox/SelectBox.tsx b/src/components/SelectBox/SelectBox.tsx new file mode 100644 index 0000000..f309538 --- /dev/null +++ b/src/components/SelectBox/SelectBox.tsx @@ -0,0 +1,45 @@ +import { useIntl } from "@cookbook/solid-intl"; +import { Select, createOptions } from "@thisbeyond/solid-select"; + +// Import default styles. (All examples use this via a global import) +import "@thisbeyond/solid-select/style.css"; +import { Component } from "solid-js"; +import { placeholders } from "../../translations"; +import { FeedOption } from "../../types/primal"; + +// Apply custom styling. See stylesheet below. +import "./SelectBox.scss"; + +const SelectBox: Component<{ options: () => FeedOption[], onChange: (value: any) => void, initialValue: any, isSelected: (value: any) => boolean, isPhone?: boolean }> = (props) => { + + const intl = useIntl(); + + const opts = createOptions(props.options, { key: 'label', disable: props.isSelected }) + + const onFocus = () => { + const control = document.querySelector('.solid-select-control'); + control?.classList.add('highlighted'); + } + + const onBlur = () => { + const control = document.querySelector('.solid-select-control'); + control?.classList.remove('highlighted'); + } + + return ( + + +
+
+ Set custom zap amount presets: +
+
+ + {(value, index) => + changeZapOptions(e, index())} + /> + } + +
+
+ + ); +} + +export default SettingsZap; diff --git a/src/components/SmallCallToAction/SmallCallToAction.module.scss b/src/components/SmallCallToAction/SmallCallToAction.module.scss new file mode 100644 index 0000000..5825fa7 --- /dev/null +++ b/src/components/SmallCallToAction/SmallCallToAction.module.scss @@ -0,0 +1,42 @@ +.callToAction { + display: grid; + // height: 32px; + min-width: 50%; + grid-template-columns: 32px 1fr; + grid-column-gap: 10px; + align-items: center; + background-color: unset; + margin: 0px; + padding: 0px; + border: none; + outline: none; + + p { + font-size: 34px; + line-height: 34px; + padding: 0px; + margin: 0px; + } + + .border { + height: 28px; + padding: 1px; + background: var(--brand-gradient); + border-radius: 6px; + } + + .input { + height: 26px; + font-size: 14px; + line-height: 20px; + font-weight: 400; + margin: 0px; + border-radius: 6px; + border: none; + color: var(--text-tertiary); + background-color: var(--background-site); + display: flex; + align-items: center; + padding-left: 12px; + } +} diff --git a/src/components/SmallCallToAction/SmallCallToAction.tsx b/src/components/SmallCallToAction/SmallCallToAction.tsx new file mode 100644 index 0000000..faa7675 --- /dev/null +++ b/src/components/SmallCallToAction/SmallCallToAction.tsx @@ -0,0 +1,35 @@ +import { Component } from 'solid-js'; +import Avatar from '../Avatar/Avatar'; + +import styles from './SmallCallToAction.module.scss'; +import { useAccountContext } from '../../contexts/AccountContext'; +import { PrimalUser } from '../../types/primal'; +import { placeholders } from '../../translations'; +import { useIntl } from '@cookbook/solid-intl'; + +const SmallCallToAction: Component<{ activeUser: PrimalUser | undefined }> = (params) => { + + const account = useAccountContext(); + const intl = useIntl(); + + const showNewNoteForm = () => { + account?.actions?.showNewNoteForm(); + }; + + return ( + + ); +} + +export default SmallCallToAction; diff --git a/src/components/SmallNote/SmallNote.module.scss b/src/components/SmallNote/SmallNote.module.scss new file mode 100644 index 0000000..3d7ed28 --- /dev/null +++ b/src/components/SmallNote/SmallNote.module.scss @@ -0,0 +1,85 @@ +.smallNote { + margin-bottom: 18px; + display: grid; + grid-template-columns: 24px 1fr; + grid-template-rows: 1fr; + grid-template-areas: "avatar content"; + grid-column-gap: 12px; + width: 300px; + + .avatar { + grid-area: avatar; + display: grid; + justify-items: center; + + .avatarImg { + width: 24px; + height: 24px; + border-radius: 50%; + } + } + + .content { + grid-area: content; + display: grid; + grid-template-columns: 1fr; + grid-template-rows: 16px 1fr; + grid-row-gap: 2px; + grid-template-areas: "header" "message"; + + .header { + grid-area: header; + font-size: 14px; + line-height: 18px; + display: flex; + align-items: center; + + .name { + color: var(--text-secondary-2); + font-weight: 800; + max-width: 120px; + text-overflow: ellipsis; + overflow: hidden; + white-space: nowrap; + } + + .time{ + margin: 0px 2px; + color: var(--text-tertiary); + font-weight: 400; + &::before { + content: "|"; + padding: 0px 2px; + } + } + } + + .message { + grid-area: message; + word-break: break-word; + font-size: 14px; + line-height: 18px; + font-weight: 400; + color: var(--text-tertiary); + width: 264px; + + overflow: hidden; + display: -webkit-box; + line-clamp: 2; + -webkit-line-clamp: 2; + -webkit-box-orient: vertical; + -moz-box-orient: vertical; + -ms-box-orient: vertical; + text-overflow: ellipsis; + } + + &:hover { + .message, .name, .time { + color: var(--text-primary); + text-decoration: underline; + } + + cursor: pointer; + } + } +} diff --git a/src/components/SmallNote/SmallNote.tsx b/src/components/SmallNote/SmallNote.tsx new file mode 100644 index 0000000..0e3e970 --- /dev/null +++ b/src/components/SmallNote/SmallNote.tsx @@ -0,0 +1,107 @@ +import { Component, JSXElement, Show } from 'solid-js'; +import Avatar from '../Avatar/Avatar'; + +import styles from './SmallNote.module.scss'; +import { A } from '@solidjs/router'; +import { PrimalNote } from '../../types/primal'; +import { useThreadContext } from '../../contexts/ThreadContext'; +import { date } from '../../lib/dates'; +import { authorName } from '../../stores/profile'; +import { note as t } from '../../translations'; +import { useIntl } from '@cookbook/solid-intl'; + + +const SmallNote: Component<{ note: PrimalNote, children?: JSXElement }> = (props) => { + + const threadContext = useThreadContext(); + const intl = useIntl(); + + const navToThread = (note: PrimalNote) => { + threadContext?.actions.setPrimaryNote(note); + }; + + const nameOfAuthor = () => { + return authorName(props.note.user || { pubkey: props.note.post.pubkey }); + }; + + const parsedContent = (text: string) => { + const regex = /\#\[([0-9]*)\]/g; + let parsed = text; + + let refs = []; + let match; + + while((match = regex.exec(text)) !== null) { + refs.push(match[1]); + } + + if (refs.length > 0) { + for(let i =0; i < refs.length; i++) { + let r = parseInt(refs[i]); + + const tag = props.note.post.tags[r]; + + if (tag[0] === 'e' && props.note.mentionedNotes && props.note.mentionedNotes[tag[1]]) { + const embeded = ( + {intl.formatMessage( + t.mentionIndication, + { + name: authorName(props.note.user), + }, + )} + ); + + // @ts-ignore + parsed = parsed.replace(`#[${r}]`, embeded.outerHTML); + } + + if (tag[0] === 'p' && props.note.mentionedUsers && props.note.mentionedUsers[tag[1]]) { + const user = props.note.mentionedUsers[tag[1]]; + + const link = (@{authorName(user)}); + + // @ts-ignore + parsed = parsed.replace(`#[${r}]`, link.outerHTML); + } + } + } + + return parsed; + + }; + + return ( + + ); +} + +export default SmallNote; diff --git a/src/components/StickySidebar/StickySidebar.module.scss b/src/components/StickySidebar/StickySidebar.module.scss new file mode 100644 index 0000000..adae438 --- /dev/null +++ b/src/components/StickySidebar/StickySidebar.module.scss @@ -0,0 +1,15 @@ +.trendingSection { + width: 300px; + margin-bottom: 228px; +} + +.stickyWrapper { + position: fixed; + height: 100%; + overflow: hidden; + overflow-y: scroll; + + &::-webkit-scrollbar{ + display: none; + } +} diff --git a/src/components/StickySidebar/StickySidebar.tsx b/src/components/StickySidebar/StickySidebar.tsx new file mode 100644 index 0000000..a742522 --- /dev/null +++ b/src/components/StickySidebar/StickySidebar.tsx @@ -0,0 +1,21 @@ +import { Component, JSXElement } from 'solid-js'; +import Wormhole from '../Wormhole/Wormhole'; + +import styles from './StickySidebar.module.scss'; + +const StickySidebar: Component<{ children: JSXElement }> = (props) => { + + return ( + + + + ); +} + +export default StickySidebar; diff --git a/src/components/ThemeChooser/ThemeChooser.module.scss b/src/components/ThemeChooser/ThemeChooser.module.scss new file mode 100644 index 0000000..53afaf8 --- /dev/null +++ b/src/components/ThemeChooser/ThemeChooser.module.scss @@ -0,0 +1,6 @@ + +.themeChooser { + display: grid; + width: 100%; + grid-template-columns: 1fr 1fr 1fr 1fr; +} diff --git a/src/components/ThemeChooser/ThemeChooser.tsx b/src/components/ThemeChooser/ThemeChooser.tsx new file mode 100644 index 0000000..c025fc8 --- /dev/null +++ b/src/components/ThemeChooser/ThemeChooser.tsx @@ -0,0 +1,31 @@ +import { Component, For } from 'solid-js'; + +import styles from './ThemeChooser.module.scss'; +import ThemeOption from './ThemeOption/ThemeOption'; +import { useSettingsContext } from '../../contexts/SettingsContext'; +import { PrimalTheme } from '../../types/primal'; + +const ThemeChooser: Component = () => { + + const settings = useSettingsContext(); + + const onSelect = (theme: PrimalTheme) => { + settings?.actions?.setTheme(theme); + }; + + return ( +
+ + {(theme) => ( + onSelect(theme)} + /> + )} + +
+ ); +} + +export default ThemeChooser; diff --git a/src/components/ThemeChooser/ThemeOption/ThemeOption.module.scss b/src/components/ThemeChooser/ThemeOption/ThemeOption.module.scss new file mode 100644 index 0000000..3a1ae2d --- /dev/null +++ b/src/components/ThemeChooser/ThemeOption/ThemeOption.module.scss @@ -0,0 +1,98 @@ + +.themeOption { + width: 128px; + >p { + font-size: 14px; + line-height: 16px; + font-weight: 500; + color: var(--text-primary); + margin-top: 8px; + text-align: center; + } +} + + +@mixin themeOptionButton { + position: relative; + width: 100%; + height: 100px; + background-color: unset; + border: none; + margin: 0px; + padding: 0px; + display: grid; + place-content: center; + border-radius: 6px; + + >img { + width: 58px; + height: 58px; + } +} + +.selected { + border: solid 1px var(--brand-1) !important; +} + +.sunset { + @include themeOptionButton(); + background-color: var(--dark-back); +} + +.midnight { + @include themeOptionButton(); + background-color: var(--dark-back); +} + +.sunrise { + @include themeOptionButton(); + background-color: var(--light-back); +} + +.ice { + @include themeOptionButton(); + background-color: var(--light-back); +} + +.themeChecked { + position: absolute; + bottom: 4px; + right: 4px; + width: 20px; + height: 20px; + display: flex; + margin: 0px; + padding: 0px; + align-items: center; + justify-content: center; + background: var(--brand-gradient); + border-radius: 50%; + >img { + width: 10px; + height: 10px; + } +} + +@mixin themeUnchecked { + position: absolute; + bottom: 4px; + right: 4px; + width: 20px; + height: 20px; + display: block; + margin: 0px; + padding: 0px; + align-items: center; + justify-content: center; + border-radius: 50%; +} + +.themeUncheckedDark { + @include themeUnchecked(); + background: var(--dark-input); +} + +.themeUncheckedLight { + @include themeUnchecked(); + background: var(--light-input); +} diff --git a/src/components/ThemeChooser/ThemeOption/ThemeOption.tsx b/src/components/ThemeChooser/ThemeOption/ThemeOption.tsx new file mode 100644 index 0000000..2f13136 --- /dev/null +++ b/src/components/ThemeChooser/ThemeOption/ThemeOption.tsx @@ -0,0 +1,40 @@ +import { Component, Show } from 'solid-js'; +import styles from './ThemeOption.module.scss'; + +import check from '../../../assets/icons/check.svg'; +import { PrimalTheme } from '../../../types/primal'; + +const ThemeOption: Component<{ + theme: PrimalTheme, + isSelected: boolean, + onSelect: (value: PrimalTheme) => void, +}> = (props) => { + + const selectedClass = () => { + return props.isSelected ? styles.selected : ''; + }; + + const uncheckedTheme = () => { + return props.theme.dark ? styles.themeUncheckedDark : styles.themeUncheckedLight; + } + + return ( +
+
} + > +
+ + +

{props.theme.label}

+ + ); +} + +export default ThemeOption; diff --git a/src/components/Toaster/Toaster.module.scss b/src/components/Toaster/Toaster.module.scss new file mode 100644 index 0000000..c695aa0 --- /dev/null +++ b/src/components/Toaster/Toaster.module.scss @@ -0,0 +1,50 @@ +.toastHolder { + position: fixed; + width: 100%; + display: flex; + justify-content: center; + align-items: center; + z-index: 100; +} + +@keyframes fadein { + from { margin-top: -70px;} + to { margin-top: 12px;} +} + +@mixin toastMessage { + min-width: 100px; + min-height: 44px; + margin: 12px; + animation: fadein 0.6s; + color: white; + font-weight: 400; + font-size: 18px; + line-height: 20px; + border-radius: 6px; + display: flex; + justify-content: center; + align-items: center; + padding-inline: 16px; +} + +.toastSuccess { + @include toastMessage(); + background: linear-gradient(0deg, rgba(41, 91, 2, 0.5), rgba(41, 91, 2, 0.5)), #000000; + border-color: 1px solid rgba(255, 255, 255, 0.1); + box-shadow: 0px 4px 8px rgba(0, 0, 0, 0.4); +} + +.toastWarning { + @include toastMessage(); + background: linear-gradient(0deg, rgba(144, 3, 3, 0.5), rgba(144, 3, 3, 0.5)), #000000;; + border: 1px solid rgba(255, 255, 255, 0.1); + box-shadow: 0px 4px 8px rgba(0, 0, 0, 0.4); +} + +.toastInfo { + @include toastMessage(); + background-color: #222222; + border: 1px solid rgba(255, 255, 255, 0.1); + box-shadow: 0px 4px 8px rgba(0, 0, 0, 0.4); +} diff --git a/src/components/Toaster/Toaster.tsx b/src/components/Toaster/Toaster.tsx new file mode 100644 index 0000000..7d3ff33 --- /dev/null +++ b/src/components/Toaster/Toaster.tsx @@ -0,0 +1,68 @@ +import styles from "./Toaster.module.scss"; + +import { Component, createContext, JSX, useContext } from "solid-js"; + +type ContextProps = { children: number | boolean | Node | JSX.ArrayElement | JSX.FunctionElement | (string & {}) | null | undefined; }; + +type ToastContextStore = { + sendWarning: (message: string) => void, + sendSuccess: (message: string) => void, + sendInfo: (message: string) => void, + notImplemented: () => void, +} + +const ToastContext = createContext(); + +const Toaster: Component = (props) => { + let toastHolder: HTMLDivElement | undefined = undefined; + + const toastMesage = (message: string, klass: string, duration = 4000) => { + const toaster = document.createElement('div'); + toaster.innerHTML = message; + toaster.classList.add(klass); + setTimeout(() => { + toastHolder?.append(toaster) + }, 0); + + setTimeout(() => { + toastHolder?.removeChild(toaster); + }, duration); + + }; + + const toastData = { + sendWarning: (message: string) => { + toastMesage(message, styles.toastWarning); + }, + sendSuccess: (message: string) => { + toastMesage(message, styles.toastSuccess, 2000); + }, + sendInfo: (message: string) => { + toastMesage(message, styles.toastInfo); + }, + notImplemented: () => { + toastMesage( + 'Feature not available in this preview release. Rest assured, we are working on it. Come back soon!', + styles.toastInfo, + 2000, + ); + }, + }; + + return ( + <> +
+
+ + {props.children} + + + ) +} + +export default Toaster; + +export function useToastContext() { return useContext(ToastContext); } diff --git a/src/components/VerificationCheck/VerificationCheck.module.scss b/src/components/VerificationCheck/VerificationCheck.module.scss new file mode 100644 index 0000000..94a090c --- /dev/null +++ b/src/components/VerificationCheck/VerificationCheck.module.scss @@ -0,0 +1,37 @@ + +.verifiedIcon { + width: 100%; + height: 100%; + display: inline-block; + margin-inline: 2px; + background-color: var(--text-tertiary-2); + -webkit-mask: url(../../assets/icons/verified.svg) no-repeat 0 / 100%; + mask: url(../../assets/icons/verified.svg) no-repeat 0 / 100%; +} + +.verificationIcon { + width: 15px; + height: 15px; + display: inline-block; + position: relative; +} + +.verifiedIconPrimal { + width: 100%; + height: 100%; + display: inline-block; + margin-inline: 2px; + background: var(--brand-gradient); + -webkit-mask: url(../../assets/icons/verified.svg) no-repeat 0 / 100%; + mask: url(../../assets/icons/verified.svg) no-repeat 0 / 100%; +} + +.whiteCheck { + width: 11px; + height: 11px; + position: absolute; + top: 2px; + left: 4px; + border-radius: 50%; + background-color: white; +} diff --git a/src/components/VerificationCheck/VerificationCheck.tsx b/src/components/VerificationCheck/VerificationCheck.tsx new file mode 100644 index 0000000..ab5203a --- /dev/null +++ b/src/components/VerificationCheck/VerificationCheck.tsx @@ -0,0 +1,39 @@ +import styles from "./VerificationCheck.module.scss"; + +import { Component, Match, Switch } from "solid-js"; +import { PrimalUser } from "../../types/primal"; + + +const VerificationCheck: Component<{ user: PrimalUser | undefined }> = (props) => { + + + const isVerifiedByPrimal = () => { + return !!props.user?.nip05 && + props.user?.nip05.endsWith('primal.net'); + } + + const isVerified = () => { + return !!props.user?.nip05 && + !props.user?.nip05.endsWith('primal.net'); + } + + return ( + <> + + +
+ +
+
+ +
+ + +
+
+
+ + ) +} + +export default VerificationCheck; diff --git a/src/components/Wormhole/Wormhole.tsx b/src/components/Wormhole/Wormhole.tsx new file mode 100644 index 0000000..268fea2 --- /dev/null +++ b/src/components/Wormhole/Wormhole.tsx @@ -0,0 +1,24 @@ +import { Component, createSignal, JSXElement, onMount, Show } from 'solid-js'; +import { Portal } from 'solid-js/web'; + +const Wormhole: Component<{children: JSXElement, to: string }> = (props) => { + + const [mounted, setMounted] = createSignal(false); + + onMount(() => { + setTimeout(() => { + // Temporary fix for Portal rendering on initial load. + setMounted(true); + }); + }); + + return ( + + + {props.children} + + + ); +} + +export default Wormhole; diff --git a/src/constants.ts b/src/constants.ts new file mode 100644 index 0000000..b5c860b --- /dev/null +++ b/src/constants.ts @@ -0,0 +1,251 @@ +import { FeedPage, } from "./types/primal"; +import logoFire from './assets/icons/logo_fire.svg'; +import logoIce from './assets/icons/logo_ice.svg'; +import { MessageDescriptor } from "@cookbook/solid-intl"; + +export const second = 1000; +export const minute = 60 * second; +export const hour = 60 * minute; +export const day = 24 * hour; +export const week = 7 * day; + +export const emptyPage: FeedPage = { + users: {}, + messages: [], + postStats: {}, + noteActions: {}, +} + +export const trendingFeed = { + name: 'Trending, my network', + hex: 'network;trending', + npub: 'trending;network', +}; + +export const themes = [ + { + name: 'sunset', + label: 'sunset wave', + logo: logoFire, + dark: true, + }, + { + name: 'sunrise', + label: 'sunrise wave', + logo: logoFire, + }, + { + name: 'midnight', + label: 'midnight wave', + logo: logoIce, + dark: true, + }, + { + name: 'ice', + label: 'ice wave', + logo: logoIce, + }, +]; + +export const minKnownProfiles: {"names": Record} = { + "names": { + "miljan": "d61f3bc5b3eb4400efdae6169a5c17cabf3246b514361de939ce4a1a0da6ef4a", + "marko": "123afae7d187ba36d6ddcd97dbf4acc59aeffe243f782592ff8f25ed579df306", + "essguess": "0b13870379cf18ae6b6d516d9f0833e0273c7a6758652a698e11f04c9c1a0d29", + "pr": "dd9b989dfe5e0840a92538f3e9f84f674e5f17ab05932efbacb4d8e6c905f302", + "marija": "b8a518a60fab9f3969b62238860f4643003b6437b75d60860dd8de34fb21c931", + "moysie": "2a55ed52ed31f85f8bdef3bdd165aa74265d82c952193d7b76fb4c76cccc4231", + "nikola": "97b988fbf4f8880493f925711e1bd806617b508fd3d28312288507e42f8a3368", + "princfilip": "29c07b40860f06df7c1ada6af2cc6b4c541b76a720542d7ee645c20c9452ffd2", + "highlights": "9a500dccc084a138330a1d1b2be0d5e86394624325d25084d3eca164e7ea698a", + "primal": "532d830dffe09c13e75e8b145c825718fc12b0003f61d61e9077721c7fff93cb", + "andi": "5fd8c6a375c431729a3b78e2080ffff0a1dc63f52e2a868a801151190a31f955", + "rockstar": "91c9a5e1a9744114c6fe2d61ae4de82629eaaa0fb52f48288093c7e7e036f832", + "qa": "88cc134b1a65f54ef48acc1df3665063d3ea45f04eab8af4646e561c5ae99079", + "jack": "82341f882b6eabcd2ba7f1ef90aad961cf074af15b9ef44a09f9d2a8fbfbe6a2", + } +}; + +export const defaultFeeds = [ +]; + +export const timeframeLabels: Record = { + latest: 'latest', + trending: 'trending', + popular: 'popular', + mostzapped: 'most zapped', +}; + +export const scopeLabels: Record = { + follows: 'my follows', + tribe: 'my tribe', + network: 'my network', + global: 'global' +}; + +export const noKey = 'no-key'; + +export enum Kind { + Metadata = 0, + Text = 1, + RecommendRelay = 2, + Contacts = 3, + EncryptedDirectMessage = 4, + EventDeletion = 5, + Repost = 6, + Reaction = 7, + ChannelCreation = 40, + ChannelMetadata = 41, + ChannelMessage = 42, + ChannelHideMessage = 43, + ChannelMuteUser = 44, + + Settings = 30_078, + + ACK = 10_000_098, + NoteStats = 10_000_100, + NetStats = 10_000_101, + LegendStats = 10_000_102, + UserStats = 10_000_105, + OldestEvent = 10_000_106, + Mentions = 10_000_107, + UserScore = 10_000_108, + Notification = 10_000_110, + Timestamp = 10_000_111, + NotificationStats = 10_000_112, + FeedRange = 10_000_113, + NoteActions = 10_000_115, + MessageStats = 10_000_117, + MesagePerSenderStats = 10_000_118, + MediaInfo = 10_000_119, + Upload = 10_000_120, + Uploaded = 10_000_121, +} + +export const relayConnectingTimeout = 5000; + +export enum NotificationType { + NEW_USER_FOLLOWED_YOU = 1,// + USER_UNFOLLOWED_YOU = 2,// + + YOUR_POST_WAS_ZAPPED = 3,// + YOUR_POST_WAS_LIKED = 4,// + YOUR_POST_WAS_REPOSTED = 5,// + YOUR_POST_WAS_REPLIED_TO = 6,// + YOU_WERE_MENTIONED_IN_POST = 7,// + YOUR_POST_WAS_MENTIONED_IN_POST = 8,// + + POST_YOU_WERE_MENTIONED_IN_WAS_ZAPPED = 101,// + POST_YOU_WERE_MENTIONED_IN_WAS_LIKED = 102,// + POST_YOU_WERE_MENTIONED_IN_WAS_REPOSTED = 103, + POST_YOU_WERE_MENTIONED_IN_WAS_REPLIED_TO = 104, + + POST_YOUR_POST_WAS_MENTIONED_IN_WAS_ZAPPED = 201, + POST_YOUR_POST_WAS_MENTIONED_IN_WAS_LIKED = 202,// + POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPOSTED = 203, + POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPLIED_TO = 204, +}; + +export const typeIcons: Record = { + [NotificationType.NEW_USER_FOLLOWED_YOU]: 'user_followed.svg', + [NotificationType.USER_UNFOLLOWED_YOU]: 'user_unfollowed.svg', + + [NotificationType.YOUR_POST_WAS_ZAPPED]: 'post_zapped.svg', + [NotificationType.YOUR_POST_WAS_LIKED]: 'post_liked.svg', + [NotificationType.YOUR_POST_WAS_REPOSTED]: 'post_reposted.svg', + [NotificationType.YOUR_POST_WAS_REPLIED_TO]: 'post_replied.svg', + + [NotificationType.YOU_WERE_MENTIONED_IN_POST]: 'mention.svg', + [NotificationType.YOUR_POST_WAS_MENTIONED_IN_POST]: 'mentioned_post.svg', + + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_ZAPPED]: 'mention_zapped.svg', + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_LIKED]: 'mention_liked.svg', + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_REPOSTED]: 'mention_reposted.svg', + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_REPLIED_TO]: 'mention_replied.svg', + + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_ZAPPED]: 'mentioned_post_zapped.svg', + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_LIKED]: 'mentioned_post_liked.svg', + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPOSTED]: 'mentioned_post_reposted.svg', + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPLIED_TO]: 'mentioned_post_replied.svg', + +} + +export const notificationTypeUserProps: Record = { + [NotificationType.NEW_USER_FOLLOWED_YOU]: 'follower', + [NotificationType.USER_UNFOLLOWED_YOU]: 'follower', + + [NotificationType.YOUR_POST_WAS_ZAPPED]: 'who_zapped_it', + [NotificationType.YOUR_POST_WAS_LIKED]: 'who_liked_it', + [NotificationType.YOUR_POST_WAS_REPOSTED]: 'who_reposted_it', + [NotificationType.YOUR_POST_WAS_REPLIED_TO]: 'who_replied_to_it', + + [NotificationType.YOU_WERE_MENTIONED_IN_POST]: 'you_were_mentioned_in', + [NotificationType.YOUR_POST_WAS_MENTIONED_IN_POST]: 'your_post_were_mentioned_in', + + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_ZAPPED]: 'who_zapped_it', + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_LIKED]: 'who_liked_it', + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_REPOSTED]: 'who_reposted_it', + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_REPLIED_TO]: 'who_replied_to_it', + + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_ZAPPED]: 'who_zapped_it', + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_LIKED]: 'who_liked_it', + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPOSTED]: 'who_reposted_it', + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPLIED_TO]: 'who_replied_to_it', + +} + +export const notificationTypeNoteProps: Record = { + // [NotificationType.NEW_USER_FOLLOWED_YOU]: 'follower', + // [NotificationType.USER_UNFOLLOWED_YOU]: 'follower', + + [NotificationType.YOUR_POST_WAS_ZAPPED]: 'your_post', + [NotificationType.YOUR_POST_WAS_LIKED]: 'your_post', + [NotificationType.YOUR_POST_WAS_REPOSTED]: 'your_post', + [NotificationType.YOUR_POST_WAS_REPLIED_TO]: 'reply', + + [NotificationType.YOU_WERE_MENTIONED_IN_POST]: 'you_were_mentioned_in', + [NotificationType.YOUR_POST_WAS_MENTIONED_IN_POST]: 'your_post_were_mentioned_in', + + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_ZAPPED]: 'post_you_were_mentioned_in', + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_LIKED]: 'post_you_were_mentioned_in', + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_REPOSTED]: 'post_you_were_mentioned_in', + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_REPLIED_TO]: 'reply', + + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_ZAPPED]: 'your_post_were_mentioned_in', + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_LIKED]: 'your_post_were_mentioned_in', + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPOSTED]: 'your_post_were_mentioned_in', + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPLIED_TO]: 'reply', + +} + +export const noteRegex = /nostr:((note|nevent)1\w+)\b|#\[(\d+)\]/g; +export const profileRegex = /nostr:((npub|nprofile)1\w+)\b|#\[(\d+)\]/g; +export const editMentionRegex = /(?:\s|^)@\`(.*?)\`/ig; + +export const medZapLimit = 1000; + + +export const defaultNotificationSettings: Record = { + NEW_USER_FOLLOWED_YOU: true, + USER_UNFOLLOWED_YOU: true, + + YOUR_POST_WAS_ZAPPED: true, + YOUR_POST_WAS_LIKED: true, + YOUR_POST_WAS_REPOSTED: true, + YOUR_POST_WAS_REPLIED_TO: true, + + YOU_WERE_MENTIONED_IN_POST: true, + YOUR_POST_WAS_MENTIONED_IN_POST: true, + + POST_YOU_WERE_MENTIONED_IN_WAS_ZAPPED: true, + POST_YOU_WERE_MENTIONED_IN_WAS_LIKED: true, + POST_YOU_WERE_MENTIONED_IN_WAS_REPOSTED: true, + POST_YOU_WERE_MENTIONED_IN_WAS_REPLIED_TO: true, + + POST_YOUR_POST_WAS_MENTIONED_IN_WAS_ZAPPED: true, + POST_YOUR_POST_WAS_MENTIONED_IN_WAS_LIKED: true, + POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPOSTED: true, + POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPLIED_TO: true, +}; + +export const emojiSearchLimit = 1; diff --git a/src/contexts/AccountContext.tsx b/src/contexts/AccountContext.tsx new file mode 100644 index 0000000..eda8048 --- /dev/null +++ b/src/contexts/AccountContext.tsx @@ -0,0 +1,396 @@ +import { createStore } from "solid-js/store"; +import { + createContext, + createEffect, + JSX, + onCleanup, + onMount, + useContext +} from "solid-js"; +import { + NostrContactsContent, + NostrEOSE, + NostrEvent, + NostrRelays, + NostrWindow, + PrimalNote, + PrimalUser, +} from '../types/primal'; +import { Kind, noKey } from "../constants"; +import { isConnected, refreshSocketListeners, removeSocketListeners, socket, subscribeTo } from "../sockets"; +import { sendContacts, sendLike } from "../lib/notes"; +import { Relay } from "nostr-tools"; +import { APP_ID } from "../App"; +import { getLikes, getProfileContactList, getUserProfiles } from "../lib/profile"; +import { getStorage, saveFollowing, saveLikes, saveRelaySettings } from "../lib/localStore"; +import { closeRelays, connectRelays } from "../lib/relays"; + +export type AccountContextStore = { + likes: string[], + relays: Relay[], + relaySettings: NostrRelays, + publicKey: string | undefined, + activeUser: PrimalUser | undefined, + showNewNoteForm: boolean, + following: string[], + followingSince: number, + hasPublicKey: () => boolean, + isKeyLookupDone: boolean, + actions: { + showNewNoteForm: () => void, + hideNewNoteForm: () => void, + setActiveUser: (user: PrimalUser) => void, + addLike: (note: PrimalNote) => Promise, + setPublicKey: (pubkey: string | undefined) => void, + addFollow: (pubkey: string) => void, + removeFollow: (pubkey: string) => void, + }, +} + +const initialData = { + likes: [], + relays: [], + relaySettings: {}, + publicKey: undefined, + activeUser: undefined, + showNewNoteForm: false, + following: [], + followingSince: 0, + isKeyLookupDone: false, +}; + +export const AccountContext = createContext(); + +export function AccountProvider(props: { children: number | boolean | Node | JSX.ArrayElement | JSX.FunctionElement | (string & {}) | null | undefined; }) { + + const setPublicKey = (pubkey: string | undefined) => { + updateStore('publicKey', () => pubkey); + updateStore('isKeyLookupDone', true); + }; + + const hasPublicKey: () => boolean = () => { + return !!store.publicKey && store.publicKey !== noKey; + }; + + const setRelaySettings = (settings: NostrRelays) => { + updateStore('relaySettings', () => ({ ...settings })); + saveRelaySettings(store.publicKey, settings) + } + + let connecting = false; + + const connectToRelays = (relaySettings: NostrRelays) => { + + if (connecting) { + return; + } + + connecting = true; + + closeRelays(store.relays, + () => { + connectRelays(relaySettings, (connected) => { + updateStore('relays', () => [ ...connected ]); + console.log('Connected relays: ', connected); + connecting = false; + }); + + }, + () => { + console.log('Failed to close all relays'); + connecting = false; + }, + ); + + }; + + let extensionAttempt = 0; + + const fetchNostrKey = async () => { + const win = window as NostrWindow; + const nostr = win.nostr; + + if (nostr === undefined) { + console.log('No WebLn extension'); + // Try again after one second if extensionAttempts are not exceeded + if (extensionAttempt < 1) { + extensionAttempt += 1; + setTimeout(fetchNostrKey, 1000); + return; + } + + updateStore('isKeyLookupDone', true); + return; + } + + try { + const key = await nostr.getPublicKey(); + + if (key === undefined) { + setTimeout(fetchNostrKey, 1000); + } + else { + setPublicKey(key); + localStorage.setItem('pubkey', key); + // getRelays(); + getUserProfiles([key], `user_profile_${APP_ID}`); + } + } catch (e: any) { + setPublicKey(noKey); + localStorage.setItem('pubkey', noKey); + console.log('error fetching public key: ', e); + } + } + + const setActiveUser = (user: PrimalUser) => { + updateStore('activeUser', () => ({...user})); + }; + + const showNewNoteForm = () => { + updateStore('showNewNoteForm', () => true); + }; + + const hideNewNoteForm = () => { + updateStore('showNewNoteForm', () => false); + }; + + const addLike = async (note: PrimalNote) => { + if (store.likes.includes(note.post.id)) { + return false; + } + + const success = await sendLike(note, store.relays); + + if (success) { + updateStore('likes', (likes) => [ ...likes, note.post.id]); + saveLikes(store.publicKey, store.likes); + } + + return success; + }; + + const updateContacts = (content: NostrContactsContent) => { + + const followingSince = content.created_at; + const tags = content.tags; + + const contacts = tags.reduce((acc, t) => { + return t[0] === 'p' ? [ ...acc, t[1] ] : acc; + }, []); + + setRelaySettings(JSON.parse(content.content || '{}')); + updateStore('following', () => contacts); + updateStore('followingSince', () => followingSince || 0); + saveFollowing(store.publicKey, contacts, followingSince || 0); + }; + + const addFollow = (pubkey: string) => { + if (!store.publicKey || store.following.includes(pubkey)) { + return; + } + + const unsub = subscribeTo(`before_follow_${APP_ID}`, async (type, subId, content) => { + if (type === 'EOSE') { + + if (!store.following.includes(pubkey)) { + const relayInfo = JSON.stringify(store.relaySettings); + const date = Math.floor((new Date()).getTime() / 1000); + const following = [...store.following, pubkey]; + + const success = await sendContacts(following, date, relayInfo, store.relays); + + if (success) { + updateStore('following', () => following); + updateStore('followingSince', () => date); + saveFollowing(store.publicKey, following, date); + } + } + + unsub(); + return; + } + + if (content && + content.kind === Kind.Contacts && + content.created_at && + content.created_at > store.followingSince + ) { + updateContacts(content); + } + }); + + getProfileContactList(store.publicKey, `before_follow_${APP_ID}`); + + } + + const removeFollow = (pubkey: string) => { + if (!store.publicKey || !store.following.includes(pubkey)) { + return; + } + + const unsub = subscribeTo(`before_unfollow_${APP_ID}`, async (type, subId, content) => { + if (type === 'EOSE') { + if (store.following.includes(pubkey)) { + const relayInfo = JSON.stringify(store.relaySettings); + const date = Math.floor((new Date()).getTime() / 1000); + const following = store.following.filter(f => f !== pubkey); + + const success = await sendContacts(following, date, relayInfo, store.relays); + + if (success) { + updateStore('following', () => following); + updateStore('followingSince', () => date); + saveFollowing(store.publicKey, following, date); + } + } + + unsub(); + return; + } + + if (content && + content.kind === Kind.Contacts && + content.created_at && + content.created_at > store.followingSince + ) { + updateContacts(content); + } + }); + + getProfileContactList(store.publicKey, `before_unfollow_${APP_ID}`); + + } + + +// EFFECTS -------------------------------------- + + onMount(() => { + setTimeout(() => { + updateStore('isKeyLookupDone', false); + fetchNostrKey(); + }, 1000); + }); + + createEffect(() => { + if (store.publicKey && store.publicKey !== noKey) { + + const storage = getStorage(store.publicKey); + + if (store.followingSince < storage.followingSince) { + updateStore('following', () => ({ ...storage.following })); + updateStore('followingSince', () => storage.followingSince); + } + + getProfileContactList(store.publicKey, `user_contacts_${APP_ID}`); + } + }); + + createEffect(() => { + if (store.publicKey && store.relays.length > 0) { + getLikes(store.publicKey, store.relays, (likes: string[]) => { + + updateStore('likes', () => [...likes]); + saveLikes(store.publicKey, likes); + }); + } + }); + + // If user has relays but none is connected, retry connecting after a delay + createEffect(() => { + if ( + Object.keys(store.relaySettings).length > 0 && + store.relays.length === 0 + ) { + setTimeout(() => { + connectToRelays(store.relaySettings); + }, 2000); + } + }) + + createEffect(() => { + if (isConnected()) { + refreshSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + } + }); + + createEffect(() => { + if (Object.keys(store.relaySettings).length > 0) { + connectToRelays(store.relaySettings); + } + }); + + onCleanup(() => { + removeSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + store.relays.forEach(relay => relay.close()) + }); + +// SOCKET HANDLERS ------------------------------ + + const onSocketClose = (closeEvent: CloseEvent) => { + const webSocket = closeEvent.target as WebSocket; + + webSocket.removeEventListener('message', onMessage); + webSocket.removeEventListener('close', onSocketClose); + }; + + const onMessage = (event: MessageEvent) => { + const message: NostrEvent | NostrEOSE = JSON.parse(event.data); + + const [type, subId, content] = message; + + if (subId === `user_profile_${APP_ID}`) { + if (content?.content) { + const user = JSON.parse(content.content); + + updateStore('activeUser', () => ({...user})); + } + return; + } + + if (subId === `user_contacts_${APP_ID}`) { + if (content && content.kind === Kind.Contacts) { + if (Object.keys(store.relaySettings).length === 0) { + setRelaySettings(JSON.parse(content.content || '{}')); + } + + if (!content.created_at || content.created_at <= store.followingSince) { + return; + } + + updateContacts(content); + } + return; + } + + }; + +// STORES --------------------------------------- + +const [store, updateStore] = createStore({ + ...initialData, + hasPublicKey, + actions: { + showNewNoteForm, + hideNewNoteForm, + setActiveUser, + addLike, + setPublicKey, + addFollow, + removeFollow, + }, +}); + + return ( + + {props.children} + + ); +} + +export function useAccountContext() { return useContext(AccountContext); } diff --git a/src/contexts/ExploreContext.tsx b/src/contexts/ExploreContext.tsx new file mode 100644 index 0000000..ce29c12 --- /dev/null +++ b/src/contexts/ExploreContext.tsx @@ -0,0 +1,371 @@ +import { nip19 } from "nostr-tools"; +import { createStore } from "solid-js/store"; +import { getEvents, getExploreFeed } from "../lib/feed"; +import { useAccountContext } from "./AccountContext"; +import { sortingPlan, convertToNotes, parseEmptyReposts, paginationPlan } from "../stores/note"; +import { Kind } from "../constants"; +import { + createContext, + createEffect, + onCleanup, + useContext +} from "solid-js"; +import { + getLegendStats, + startListeningForNostrStats, + stopListeningForNostrStats +} from "../lib/stats"; +import { + isConnected, + refreshSocketListeners, + removeSocketListeners, + socket +} from "../sockets"; +import { + ContextChildren, + FeedPage, + NostrEOSE, + NostrEvent, + NostrEventContent, + NostrMentionContent, + NostrNoteActionsContent, + NostrNoteContent, + NostrStatsContent, + NostrUserContent, + NoteActions, + PrimalNote, +} from "../types/primal"; +import { APP_ID } from "../App"; + +export type ExploreContextStore = { + notes: PrimalNote[], + scope: string, + timeframe: string, + isFetching: boolean, + page: FeedPage, + lastNote: PrimalNote | undefined, + reposts: Record | undefined, + isNetStatsStreamOpen: boolean, + stats: { + users: number, + pubkeys: number, + pubnotes: number, + reactions: number, + reposts: number, + any: number, + zaps: number, + satszapped: number, + }, + legend: { + your_follows: number, + your_inner_network: number, + your_outer_network: number, + }, + actions: { + saveNotes: (newNotes: PrimalNote[]) => void, + clearNotes: () => void, + fetchNotes: (topic: string, until?: number, limit?: number) => void, + fetchNextPage: () => void, + updatePage: (content: NostrEventContent) => void, + savePage: (page: FeedPage) => void, + openNetStatsStream: () => void, + closeNetStatsStream: () => void, + fetchLegendStats: (pubkey?: string) => void, + } +} + +export const initialExploreData = { + notes: [], + isFetching: false, + scope: 'global', + timeframe: 'latest', + page: { + messages: [], + users: {}, + postStats: {}, + mentions: {}, + noteActions: {}, + }, + reposts: {}, + lastNote: undefined, + isNetStatsStreamOpen: false, + stats: { + users: 0, + pubkeys: 0, + pubnotes: 0, + reactions: 0, + reposts: 0, + any: 0, + zaps: 0, + satszapped: 0, + }, + legend: { + your_follows: 0, + your_inner_network: 0, + your_outer_network: 0, + }, +}; + + +export const ExploreContext = createContext(); + +export const ExploreProvider = (props: { children: ContextChildren }) => { + + const account = useAccountContext(); + +// ACTIONS -------------------------------------- + + const saveNotes = (newNotes: PrimalNote[]) => { + + updateStore('notes', (notes) => [ ...notes, ...newNotes ]); + updateStore('isFetching', () => false); + }; + + const fetchNotes = (topic: string, until = 0, limit = 20) => { + const [scope, timeframe] = topic.split(';'); + + + if (scope && timeframe) { + updateStore('isFetching', true); + updateStore('page', () => ({ messages: [], users: {}, postStats: {} })); + + updateStore('scope', () => scope); + updateStore('timeframe', () => timeframe); + + getExploreFeed( + account?.publicKey || '', + `explore_${APP_ID}`, + scope, + timeframe, + until, + limit, + ); + return; + } + } + + const clearNotes = () => { + updateStore('page', () => ({ messages: [], users: {}, postStats: {}, noteActions: {} })); + updateStore('notes', () => []); + updateStore('reposts', () => undefined); + updateStore('lastNote', () => undefined); + }; + + const fetchNextPage = () => { + const lastNote = store.notes[store.notes.length - 1]; + + if (!lastNote) { + return; + } + + updateStore('lastNote', () => ({ ...lastNote })); + + const criteria = paginationPlan(store.timeframe); + + const noteData: Record = lastNote.repost ? + lastNote.repost.note : + lastNote.post; + + const until = noteData[criteria]; + + if (until > 0) { + fetchNotes( + `${store.scope};${store.timeframe}`, + until, + ); + } + }; + + const updatePage = (content: NostrEventContent) => { + if (content.kind === Kind.Metadata) { + const user = content as NostrUserContent; + + updateStore('page', 'users', + (usrs) => ({ ...usrs, [user.pubkey]: { ...user } }) + ); + return; + } + + if ([Kind.Text, Kind.Repost].includes(content.kind)) { + const message = content as NostrNoteContent; + + if (store.lastNote?.post?.noteId !== nip19.noteEncode(message.id)) { + updateStore('page', 'messages', + (msgs) => [ ...msgs, { ...message }] + ); + } + + return; + } + + if (content.kind === Kind.NoteStats) { + const statistic = content as NostrStatsContent; + const stat = JSON.parse(statistic.content); + + updateStore('page', 'postStats', + (stats) => ({ ...stats, [stat.event_id]: { ...stat } }) + ); + return; + } + + if (content.kind === Kind.Mentions) { + const mentionContent = content as NostrMentionContent; + const mention = JSON.parse(mentionContent.content); + + updateStore('page', 'mentions', + (mentions) => ({ ...mentions, [mention.id]: { ...mention } }) + ); + return; + } + + if (content.kind === Kind.NoteActions) { + const noteActionContent = content as NostrNoteActionsContent; + const noteActions = JSON.parse(noteActionContent.content) as NoteActions; + + updateStore('page', 'noteActions', + (actions) => ({ ...actions, [noteActions.event_id]: { ...noteActions } }) + ); + return; + } + }; + + const savePage = (page: FeedPage) => { + const sort = sortingPlan(store.timeframe); + + const newPosts = sort(convertToNotes(page)); + + saveNotes(newPosts); + }; + + const openNetStatsStream = () => { + startListeningForNostrStats(APP_ID); + }; + + const closeNetStatsStream = () => { + stopListeningForNostrStats(APP_ID); + }; + + const fetchLegendStats = (pubkey?: string) => { + if (!pubkey) { + return; + } + + getLegendStats(pubkey, APP_ID); + }; + +// SOCKET HANDLERS ------------------------------ + + const onMessage = (event: MessageEvent) => { + const message: NostrEvent | NostrEOSE = JSON.parse(event.data); + + const [type, subId, content] = message; + + if (subId === `explore_${APP_ID}`) { + if (type === 'EOSE') { + const reposts = parseEmptyReposts(store.page); + const ids = Object.keys(reposts); + + if (ids.length === 0) { + savePage(store.page); + return; + } + + updateStore('reposts', () => reposts); + + getEvents(account?.publicKey, ids, `explore_reposts_${APP_ID}`); + + return; + } + + if (type === 'EVENT') { + updatePage(content); + return; + } + } + + if ([`netstats_${APP_ID}`, `legendstats_${APP_ID}`].includes(subId) && content?.content) { + const stats = JSON.parse(content.content); + + if (content.kind === Kind.NetStats) { + updateStore('stats', () => ({ ...stats })); + } + + if (content.kind === Kind.LegendStats) { + updateStore('legend', () => ({ ...stats })); + } + } + + if (subId === `explore_reposts_${APP_ID}`) { + if (type === 'EOSE') { + savePage(store.page); + return; + } + + if (type === 'EVENT') { + const repostId = (content as NostrNoteContent).id; + const reposts = store.reposts || {}; + const parent = store.page.messages.find(m => m.id === reposts[repostId]); + + if (parent) { + updateStore('page', 'messages', (msg) => msg.id === parent.id, 'content', () => JSON.stringify(content)); + } + + return; + } + } + }; + + const onSocketClose = (closeEvent: CloseEvent) => { + const webSocket = closeEvent.target as WebSocket; + + removeSocketListeners( + webSocket, + { message: onMessage, close: onSocketClose }, + ); + }; + +// EFFECTS -------------------------------------- + + createEffect(() => { + if (isConnected()) { + refreshSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + } + }); + + onCleanup(() => { + removeSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + }); + +// STORES --------------------------------------- + + const [store, updateStore] = createStore({ + ...initialExploreData, + actions: { + saveNotes, + fetchNotes, + clearNotes, + fetchNextPage, + updatePage, + savePage, + openNetStatsStream, + closeNetStatsStream, + fetchLegendStats, + }, + }); + +// RENDER --------------------------------------- + + return ( + + {props.children} + + ); +} + +export const useExploreContext = () => useContext(ExploreContext); diff --git a/src/contexts/HomeContext.tsx b/src/contexts/HomeContext.tsx new file mode 100644 index 0000000..59c3bd6 --- /dev/null +++ b/src/contexts/HomeContext.tsx @@ -0,0 +1,522 @@ +import { nip19 } from "nostr-tools"; +import { createContext, createEffect, onCleanup, useContext } from "solid-js"; +import { createStore } from "solid-js/store"; +import { APP_ID } from "../App"; +import { Kind } from "../constants"; +import { getEvents, getExploreFeed, getFeed, getFutureExploreFeed, getFutureFeed } from "../lib/feed"; +import { searchContent } from "../lib/search"; +import { isConnected, refreshSocketListeners, removeSocketListeners, socket } from "../sockets"; +import { sortingPlan, convertToNotes, parseEmptyReposts, paginationPlan } from "../stores/note"; +import { + ContextChildren, + FeedPage, + NostrEOSE, + NostrEvent, + NostrEventContent, + NostrMentionContent, + NostrNoteActionsContent, + NostrNoteContent, + NostrStatsContent, + NostrUserContent, + NoteActions, + PrimalFeed, + PrimalNote, +} from "../types/primal"; +import { useAccountContext } from "./AccountContext"; +import { useSettingsContext } from "./SettingsContext"; + +type HomeContextStore = { + notes: PrimalNote[], + isFetching: boolean, + scrollTop: number, + selectedFeed: PrimalFeed | undefined, + page: FeedPage, + lastNote: PrimalNote | undefined, + reposts: Record | undefined, + mentionedNotes: Record, + future: { + notes: PrimalNote[], + page: FeedPage, + reposts: Record | undefined, + }, + actions: { + saveNotes: (newNotes: PrimalNote[]) => void, + clearNotes: () => void, + fetchNotes: (topic: string, subId: string, until?: number) => void, + fetchNextPage: () => void, + selectFeed: (feed: PrimalFeed | undefined) => void, + updateScrollTop: (top: number) => void, + updatePage: (content: NostrEventContent) => void, + savePage: (page: FeedPage) => void, + checkForNewNotes: (topic: string | undefined) => void, + loadFutureContent: () => void, + } +} + +const initialHomeData = { + notes: [], + isFetching: false, + scrollTop: 0, + selectedFeed: undefined, + page: { + messages: [], + users: {}, + postStats: {}, + mentions: {}, + noteActions: {}, + }, + reposts: {}, + lastNote: undefined, + mentionedNotes: {}, + future: { + notes: [], + reposts: {}, + page: { + messages: [], + users: {}, + postStats: {}, + mentions: {}, + noteActions: {}, + }, + }, +}; + +export const HomeContext = createContext(); + +export const HomeProvider = (props: { children: ContextChildren }) => { + + const settings = useSettingsContext(); + const account = useAccountContext(); + +// ACTIONS -------------------------------------- + + const clearFuture = () => { + updateStore('future', () => ({ + notes: [], + reposts: {}, + page: { + messages: [], + users: {}, + postStats: {}, + mentions: {}, + noteActions: {}, + }, + })) + } + + const saveNotes = (newNotes: PrimalNote[], scope?: 'future') => { + if (scope) { + updateStore(scope, 'notes', (notes) => [ ...notes, ...newNotes ]); + return; + } + updateStore('notes', (notes) => [ ...notes, ...newNotes ]); + updateStore('isFetching', () => false); + }; + + const checkForNewNotes = (topic: string | undefined) => { + + if (!topic) { + return; + } + + if (store.future.notes.length > 100) { + return; + } + + const [scope, timeframe] = topic.split(';'); + + let since = 0; + + if (store.notes[0]) { + since = store.notes[0].repost ? + store.notes[0].repost.note.created_at : + store.notes[0].post.created_at; + } + + // if (store.future.notes[0]) { + // since = store.future.notes[0].post.created_at; + // } + + clearFuture(); + + + if (scope && timeframe) { + + // if (scope === 'search') { + // searchFutureContent(`home_future_${APP_ID}`, decodeURI(timeframe), since); + // return; + // } + + getFutureExploreFeed( + account?.publicKey, + `home_future_${APP_ID}`, + scope, + timeframe, + since, + ); + return; + } + + getFutureFeed(account?.publicKey, topic, `home_future_${APP_ID}`, since); + } + + const loadFutureContent = () => { + if (store.future.notes.length === 0) { + return; + } + + updateStore('notes', (notes) => [...store.future.notes, ...notes]); + clearFuture(); + }; + + const fetchNotes = (topic: string, subId: string, until = 0) => { + const [scope, timeframe] = topic.split(';'); + + updateStore('isFetching', true); + updateStore('page', () => ({ messages: [], users: {}, postStats: {} })); + + if (scope && timeframe) { + + if (scope === 'search') { + searchContent(`home_feed_${subId}`, decodeURI(timeframe)); + return; + } + + getExploreFeed( + account?.publicKey, + `home_feed_${subId}`, + scope, + timeframe, + until, + ); + return; + } + + getFeed(account?.publicKey, topic, `home_feed_${subId}`, until); + }; + + const clearNotes = () => { + updateStore('scrollTop', () => 0); + updateStore('page', () => ({ messages: [], users: {}, postStats: {}, noteActions: {} })); + updateStore('notes', () => []); + updateStore('reposts', () => undefined); + updateStore('lastNote', () => undefined); + + clearFuture(); + }; + + const fetchNextPage = () => { + const lastNote = store.notes[store.notes.length - 1]; + + if (!lastNote) { + return; + } + + updateStore('lastNote', () => ({ ...lastNote })); + + const topic = store.selectedFeed?.hex; + + if (!topic) { + return; + } + + const [scope, timeframe] = topic.split(';'); + + if (scope === 'search') { + return; + } + + const pagCriteria = timeframe || 'latest'; + + const criteria = paginationPlan(pagCriteria); + + const noteData: Record = lastNote.repost ? + lastNote.repost.note : + lastNote.post; + + const until = noteData[criteria]; + + if (until > 0) { + fetchNotes(topic, `${APP_ID}`, until); + } + }; + + const updateScrollTop = (top: number) => { + updateStore('scrollTop', () => top); + }; + + const selectFeed = (feed: PrimalFeed | undefined) => { + if (feed !== undefined && feed.hex !== undefined) { + updateStore('selectedFeed', () => ({ ...feed })); + clearNotes(); + fetchNotes(feed.hex , `${APP_ID}`); + } + }; + + const updatePage = (content: NostrEventContent, scope?: 'future') => { + if (content.kind === Kind.Metadata) { + const user = content as NostrUserContent; + + if (scope) { + updateStore(scope, 'page', 'users', + (usrs) => ({ ...usrs, [user.pubkey]: { ...user } }) + ); + return; + } + + updateStore('page', 'users', + (usrs) => ({ ...usrs, [user.pubkey]: { ...user } }) + ); + return; + } + + if ([Kind.Text, Kind.Repost].includes(content.kind)) { + const message = content as NostrNoteContent; + const messageId = nip19.noteEncode(message.id); + + if (scope) { + const isFirstNote = message.kind === Kind.Text ? + store.notes[0]?.post?.noteId === messageId : + store.notes[0]?.repost?.note.noteId === messageId; + + // const isAlreadyFetched = message.kind === Kind.Text ? + // store.future.notes[0]?.post?.noteId === messageId : + // store.future.notes[0]?.repost?.note.noteId === messageId; + + if (!isFirstNote) { + updateStore(scope, 'page', 'messages', + (msgs) => [ ...msgs, { ...message }] + ); + } + return; + } + + const isLastNote = message.kind === Kind.Text ? + store.lastNote?.post?.noteId === messageId : + store.lastNote?.repost?.note.noteId === messageId; + + if (!isLastNote) { + updateStore('page', 'messages', + (msgs) => [ ...msgs, { ...message }] + ); + } + + return; + } + + if (content.kind === Kind.NoteStats) { + const statistic = content as NostrStatsContent; + const stat = JSON.parse(statistic.content); + + if (scope) { + updateStore(scope, 'page', 'postStats', + (stats) => ({ ...stats, [stat.event_id]: { ...stat } }) + ); + return; + } + updateStore('page', 'postStats', + (stats) => ({ ...stats, [stat.event_id]: { ...stat } }) + ); + return; + } + + if (content.kind === Kind.Mentions) { + const mentionContent = content as NostrMentionContent; + const mention = JSON.parse(mentionContent.content); + + if (scope) { + updateStore(scope, 'page', 'mentions', + (mentions) => ({ ...mentions, [mention.id]: { ...mention } }) + ); + return; + } + + updateStore('page', 'mentions', + (mentions) => ({ ...mentions, [mention.id]: { ...mention } }) + ); + return; + } + + if (content.kind === Kind.NoteActions) { + const noteActionContent = content as NostrNoteActionsContent; + const noteActions = JSON.parse(noteActionContent.content) as NoteActions; + + if (scope) { + updateStore(scope, 'page', 'noteActions', + (actions) => ({ ...actions, [noteActions.event_id]: { ...noteActions } }) + ); + return; + } + + updateStore('page', 'noteActions', + (actions) => ({ ...actions, [noteActions.event_id]: { ...noteActions } }) + ); + return; + } + }; + + const savePage = (page: FeedPage, scope?: 'future') => { + const topic = (store.selectedFeed?.hex || '').split(';'); + const sortingFunction = sortingPlan(topic[1]); + + const newPosts = sortingFunction(convertToNotes(page)); + + saveNotes(newPosts, scope); + }; + +// SOCKET HANDLERS ------------------------------ + + const onMessage = (event: MessageEvent) => { + const message: NostrEvent | NostrEOSE = JSON.parse(event.data); + + const [type, subId, content] = message; + + if (subId === `home_feed_${APP_ID}`) { + if (type === 'EOSE') { + const reposts = parseEmptyReposts(store.page); + const ids = Object.keys(reposts); + + if (ids.length === 0) { + savePage(store.page); + return; + } + + updateStore('reposts', () => reposts); + + getEvents(account?.publicKey, ids, `home_reposts_${APP_ID}`); + + return; + } + + if (type === 'EVENT') { + updatePage(content); + return; + } + } + + if (subId === `home_reposts_${APP_ID}`) { + if (type === 'EOSE') { + savePage(store.page); + return; + } + + if (type === 'EVENT') { + const repostId = (content as NostrNoteContent).id; + const reposts = store.reposts || {}; + const parent = store.page.messages.find(m => m.id === reposts[repostId]); + + if (parent) { + updateStore('page', 'messages', (msg) => msg.id === parent.id, 'content', () => JSON.stringify(content)); + } + + return; + } + } + + if (subId === `home_future_${APP_ID}`) { + if (type === 'EOSE') { + const reposts = parseEmptyReposts(store.future.page); + const ids = Object.keys(reposts); + + if (ids.length === 0) { + savePage(store.future.page, 'future'); + return; + } + + updateStore('future', 'reposts', () => reposts); + + getEvents(account?.publicKey, ids, `home_future_reposts_${APP_ID}`); + + return; + } + + if (type === 'EVENT') { + updatePage(content, 'future'); + return; + } + } + + if (subId === `home_future_reposts_${APP_ID}`) { + if (type === 'EOSE') { + savePage(store.future.page, 'future'); + return; + } + + if (type === 'EVENT') { + const repostId = (content as NostrNoteContent).id; + const reposts = store.future.reposts || {}; + const parent = store.future.page.messages.find(m => m.id === reposts[repostId]); + + if (parent) { + updateStore('future', 'page', 'messages', (msg) => msg.id === parent.id, 'content', () => JSON.stringify(content)); + } + + return; + } + } + + + }; + + const onSocketClose = (closeEvent: CloseEvent) => { + const webSocket = closeEvent.target as WebSocket; + + removeSocketListeners( + webSocket, + { message: onMessage, close: onSocketClose }, + ); + }; + +// EFFECTS -------------------------------------- + + createEffect(() => { + if (isConnected()) { + refreshSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + } + }); + + createEffect(() => { + if (account?.isKeyLookupDone) { + selectFeed(settings?.defaultFeed); + } + }); + + onCleanup(() => { + removeSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + }); + + +// STORES --------------------------------------- + + const [store, updateStore] = createStore({ + ...initialHomeData, + actions: { + saveNotes, + clearNotes, + fetchNotes, + fetchNextPage, + selectFeed, + updateScrollTop, + updatePage, + savePage, + checkForNewNotes, + loadFutureContent, + }, + }); + +// RENDER ------------------------------------- + + return ( + + {props.children} + + ); +} + +export const useHomeContext = () => useContext(HomeContext); diff --git a/src/contexts/MediaContext.tsx b/src/contexts/MediaContext.tsx new file mode 100644 index 0000000..1cbe014 --- /dev/null +++ b/src/contexts/MediaContext.tsx @@ -0,0 +1,116 @@ +import { createStore } from "solid-js/store"; +import { + createContext, + createEffect, + JSXElement, + onCleanup, + useContext +} from "solid-js"; +import { MediaEvent, MediaSize, MediaVariant, NostrEOSE, NostrEvent } from "../types/primal"; +import { removeSocketListeners, isConnected, refreshSocketListeners, socket } from "../sockets"; +import { Kind } from "../constants"; + +export type MediaContextStore = { + media: Record, + actions: { + getMedia: (url: string , size?: MediaSize, animated?: boolean) => MediaVariant | undefined, + getMediaUrl: (url: string | undefined, size?: MediaSize, animated?: boolean) => string | undefined, + }, +} + +const initialData = { + media: {}, +}; + +export const MediaContext = createContext(); + +export const MediaProvider = (props: { children: JSXElement }) => { + + const getMedia = (url: string, size?: MediaSize , animated?: boolean) => { + const variants: MediaVariant[] = store.media[url] || []; + + const isOfSize = (s: MediaSize) => size ? size === s : true; + const isAnimated = (a: 0 | 1) => animated !== undefined ? animated === !!a : true; + + return variants.find(v => isOfSize(v.s) && isAnimated(v.a)); + }; + + const getMediaUrl = (url: string | undefined, size?: MediaSize, animated?: boolean) => { + if (!url) { + return; + } + + const media = getMedia(url, size, animated); + + return media?.media_url; + } + +// SOCKET HANDLERS ------------------------------ + + const onMessage = (event: MessageEvent) => { + const message: NostrEvent | NostrEOSE = JSON.parse(event.data); + + const [type, _, content] = message; + + if (type === 'EVENT') { + if (content.kind === Kind.MediaInfo) { + const mediaInfo: MediaEvent = JSON.parse(content.content); + + let media: Record = {}; + + for (let i = 0;i ({ ...media })); + } + } + }; + + const onSocketClose = (closeEvent: CloseEvent) => { + const webSocket = closeEvent.target as WebSocket; + + removeSocketListeners( + webSocket, + { message: onMessage, close: onSocketClose }, + ); + }; + +// EFFECTS -------------------------------------- + + createEffect(() => { + if (isConnected()) { + refreshSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + } + }); + + onCleanup(() => { + removeSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + }); + + +// STORES --------------------------------------- + + const [store, updateStore] = createStore({ + ...initialData, + actions: { + getMedia, + getMediaUrl, + }, + }); + + return ( + + {props.children} + + ); +} + +export const useMediaContext = () => useContext(MediaContext); diff --git a/src/contexts/MessagesContext.tsx b/src/contexts/MessagesContext.tsx new file mode 100644 index 0000000..2e96743 --- /dev/null +++ b/src/contexts/MessagesContext.tsx @@ -0,0 +1,816 @@ +import { createStore } from "solid-js/store"; +import { Kind } from "../constants"; +import { + createContext, + createEffect, + onCleanup, + useContext +} from "solid-js"; +import { + isConnected, + refreshSocketListeners, + removeSocketListeners, + socket, + subscribeTo +} from "../sockets"; +import { + ContextChildren, + FeedPage, + NostrEOSE, + NostrEvent, + NostrMentionContent, + NostrMessageEncryptedContent, + NostrNoteActionsContent, + NostrNoteContent, + NostrStatsContent, + NostrUserContent, + NostrWindow, + NoteActions, + PrimalNote, + PrimalUser, + UserRelation, +} from "../types/primal"; +import { APP_ID } from "../App"; +import { getMessageCounts, getNewMessages, getOldMessages, markAllAsRead, resetMessageCount, subscribeToMessagesStats } from "../lib/messages"; +import { useAccountContext } from "./AccountContext"; +import { convertToUser } from "../stores/profile"; +import { getUserProfiles } from "../lib/profile"; +import { getEvents } from "../lib/feed"; +import { nip19 } from "nostr-tools"; +import { convertToNotes } from "../stores/note"; +import { sanitize, sendEvent } from "../lib/notes"; + + +type DirectMessage = { + id: string, + sender: string, + content: string, + created_at: number, +}; + +type DirectMessageThread = { + author: string, + messages: DirectMessage[], +}; + +type SenderMessageCount = { + cnt: number, + latest_at: number, + latest_event_id: string, +} + +export type MessagesContextStore = { + messageCount: number, + messageCountPerSender: Record, + senders: Record; + selectedSender: PrimalUser | null, + encryptedMessages: NostrMessageEncryptedContent[], + messages: DirectMessage[], + conversation: DirectMessageThread[], + isConversationLoaded: boolean, + referecedUsers: Record, + referecedNotes: Record, + referencePage: FeedPage, + now: number, + senderRelation: UserRelation, + addSender: PrimalUser | undefined, + actions: { + getMessagesPerSender: () => void, + changeSenderRelation: (relation: UserRelation) => void, + selectSender: (senderId: string | undefined) => void, + resetConversationLoaded: () => void, + addToConversation: (messages: DirectMessage[]) => void, + sendMessage: (receiver: PrimalUser, message: DirectMessage) => Promise, + resetAllMessages: () => Promise, + addSender: (user: PrimalUser) => void, + getNextConversationPage: () => void, + addUserReference: (user: PrimalUser) => void, + } +} + +export const initialData = { + messageCount: 0, + messageCountPerSender: {}, + senders: {}, + selectedSender: null, + encryptedMessages: [], + messages: [], + conversation: [], + isConversationLoaded: false, + referecedUsers: {}, + referecedNotes: {}, + now: Math.floor(new Date().getTime() / 1000), + senderRelation: 'follows' as UserRelation, + addSender: undefined, + referencePage: { + messages: [], + users: {}, + postStats: {}, + mentions: {}, + noteActions: {}, + }, +}; + + +export const MessagesContext = createContext(); + +export const MessagesProvider = (props: { children: ContextChildren }) => { + + const account = useAccountContext(); + + const subidMsgCount = `msg_stats_${APP_ID}`; + const subidMsgCountPerSender = `msg_count_p_s_ ${APP_ID}`; + const subidResetMsgCount = `msg_reset_ ${APP_ID}`; + const subidResetMsgCounts = `msg_mark_as_read_${APP_ID}`; + const subidCoversation = `msg_conv_ ${APP_ID}`; + const subidCoversationNextPage = `msg_conv_np_ ${APP_ID}`; + const subidNewMsg = `msg_new_ ${APP_ID}`; + const subidNoteRef = `msg_note_ ${APP_ID}`; + const subidUserRef = `msg_user_ ${APP_ID}`; + + + const getNostr = () => { + const win = window as NostrWindow; + return win.nostr; + } + +// ACTIONS -------------------------------------- + + const changeSenderRelation = (relation: UserRelation) => { + updateStore('senderRelation', () => relation); + // @ts-ignore + updateStore('senders', () => undefined ); + updateStore('senders', () => ({})); + getMessagesPerSender(true); + }; + + const subToMessagesStats = () => { + if (!account?.hasPublicKey()) { + return; + } + + // @ts-ignore + subscribeToMessagesStats(account?.publicKey, subidMsgCount); + } + + const getMessagesPerSender = (changeSender?: boolean) => { + if (account?.isKeyLookupDone && account.hasPublicKey()) { + changeSender && updateStore('selectedSender', () => null); + // @ts-ignore + getMessageCounts(account.publicKey, store.senderRelation, subidMsgCountPerSender); + } + }; + + const selectSender = async (senderId: string | undefined) => { + if (!senderId) { + return; + } + + let pubkey = senderId; + + if (senderId.startsWith('npub') || senderId.startsWith('nevent')) { + const decoded = nip19.decode(senderId); + + if (decoded.type === 'npub') { + pubkey = decoded.data; + } + + if (decoded.type === 'nevent') { + pubkey = decoded.data.id; + } + } + + if (!store.senders) { + return; + } + + const sender = store.senders[pubkey]; + + if (!sender) { + findMissingUser(pubkey); + return; + } + + await resetMessageCount(sender.pubkey, subidResetMsgCount); + + updateStore('selectedSender', () => null); + updateStore('selectedSender', () => ({ ...sender })); + }; + + const findMissingUser = (pubkey: string) => { + const subid = `msg_unk_${APP_ID}`; + let user: PrimalUser | undefined; + + const unsub = subscribeTo(subid, (type, subId, content) => { + + if (type === 'EVENT') { + if (content?.kind === Kind.Metadata) { + user = convertToUser(content); + } + } + + if (type === 'EOSE') { + user && addSender(user); + unsub(); + } + }); + + getUserProfiles([pubkey], subid); + }; + + const resetAllMessages = async () => { + markAllAsRead(subidResetMsgCounts); + }; + + const getConversationWithSender = (sender: PrimalUser | null, until = 0) => { + if (!account?.isKeyLookupDone || !account.hasPublicKey() || !sender) { + return; + } + resetConversationLoaded(); + // @ts-ignore + getOldMessages(account.publicKey, sender.pubkey, subidCoversation, until); + }; + + const getNextConversationPage = () => { + if ( + !account?.isKeyLookupDone || + !account.hasPublicKey() || + !store.selectedSender + ) { + return; + + } + const lastMessage = store.messages[store.messages.length - 1] || { created_at: 0}; + + updateStore('encryptedMessages', () => []); + + // @ts-ignore + lastMessage.created_at > 0 && getOldMessages(account.publicKey, store.selectedSender.pubkey, subidCoversationNextPage, lastMessage.created_at); + }; + + const decryptMessages = async (then: (messages: DirectMessage[]) => void) => { + const nostr = getNostr(); + + if (nostr === undefined || store.selectedSender === null) { + return; + } + + let newMessages: DirectMessage[] = []; + + for (let i = 0; i < store.encryptedMessages.length; i++) { + const eMsg = store.encryptedMessages[i]; + + if (!store.messages.find(m => eMsg.id === m.id) && store.selectedSender) { + try { + const content = await nostr.nip04.decrypt(store.selectedSender.pubkey, eMsg.content); + + const msg: DirectMessage = { + sender: eMsg.pubkey, + content: sanitize(content), + created_at: eMsg.created_at, + id: eMsg.id, + }; + + newMessages.push(msg); + } catch (e) { + console.warn('Falied to decrypt message: ', e); + } + } + } + + updateStore('messages', (conv) => [ ...conv, ...newMessages ]); + resetMessageCount(store.selectedSender.pubkey, subidResetMsgCount); + updateStore('messageCountPerSender', store.selectedSender.pubkey, 'cnt', 0) + + parseForMentions(newMessages); + then(newMessages); + // areNewMessages ? addToConversation(newMessages, true) : generateConversation(newMessages); + }; + + const parseForMentions = (messages: DirectMessage[]) => { + const noteRegex = /\bnostr:((note|nevent)1\w+)\b|#\[(\d+)\]/g; + const userRegex = /\bnostr:((npub|nprofile)1\w+)\b|#\[(\d+)\]/g; + + let noteRefs = []; + let userRefs = []; + let match; + + for (let i=0; i { + const decoded = nip19.decode(x); + + if (decoded.type === 'npub') { + return decoded.data; + } + + if (decoded.type === 'nprofile') { + return decoded.data.pubkey; + } + + return ''; + + }); + const noteIds = noteRefs.map(x => { + const decoded = nip19.decode(x); + + if (decoded.type === 'note') { + return decoded.data; + } + + if (decoded.type === 'nevent') { + return decoded.data.id; + } + + return ''; + + }); + + updateStore('referencePage', () => ({ + messages: [], + users: {}, + postStats: {}, + mentions: {}, + noteActions: {}, + })); + + getUserProfiles(pubkeys, subidUserRef); + getEvents(account?.publicKey, noteIds, subidNoteRef, true); + + + }; + + const prependToConversation = (messages: DirectMessage[]) => { + let firstThread = store.conversation[store.conversation.length - 1]; + + for (let i=0;i [...msgs, message] + ); + } + else { + firstThread = { + author: message.sender, + messages: [message], + } + + updateStore('conversation', (conv) => [...conv, { ...firstThread }]); + } + + // updateStore('isConversationLoaded', () => true); + updateMessageTimings(); + + }; + + }; + + const addToConversation = (messages: DirectMessage[], ignoreMy?: boolean) => { + let lastThread = store.conversation[0]; + + for (let i=0;i [message, ...msgs] + ); + } + else { + lastThread = { + author: message.sender, + messages: [message], + } + + updateStore('conversation', (conv) => [{ ...lastThread }, ...conv]); + } + + updateStore('isConversationLoaded', () => true); + updateMessageTimings(); + + }; + }; + + const generateConversation = (messages: DirectMessage[]) => { + + let author: string | undefined; + let thread: DirectMessageThread = { author: '', messages: [] }; + let conversation: any[] = []; + + for (let i=0;i 0 && Math.abs(thread.messages[thread.messages.length - 1].created_at - message.created_at) > 900 + )) { + author = message.sender; + thread.messages.length > 0 && conversation.push(thread); + thread = { author, messages: []}; + } + + thread.messages.push(message); + + }; + + thread.messages.length > 0 && conversation.push(thread); + + updateStore('conversation', (conv) => [...conv, ...conversation]); + updateStore('isConversationLoaded', () => true); + }; + + const resetConversationLoaded = () => { + updateStore('isConversationLoaded', () => false); + } + + const updateRefUsers = () => { + const refs = store.referencePage.users; + + const users = Object.keys(refs).reduce((acc, id) => { + const user = convertToUser(refs[id]); + return {...acc, [user.pubkey]: { ...user }}; + }, {}); + + updateStore('referecedUsers', (usrs) => ({ ...usrs, ...users })); + }; + + const updateRefNotes = () => { + const refs = convertToNotes(store.referencePage) || []; + + const notes = refs.reduce((acc, note) => { + return { ...acc, [note.post.noteId]: note }; + }, {}); + + updateStore('referecedNotes', (nts) => ({ ...nts, ...notes })); + }; + + const sendMessage = async (receiver: PrimalUser, message: DirectMessage) => { + const nostr = getNostr(); + if (!account || !nostr) { + return false; + } + + const content = await nostr.nip04.encrypt(receiver.pubkey, message.content); + + const event = { + content, + kind: Kind.EncryptedDirectMessage, + tags: [['p', receiver.pubkey]], + created_at: Math.floor((new Date).getTime() / 1000), + }; + + const success = await sendEvent(event, account?.relays); + + if (success) { + const msg = { ...message, content: sanitize(message.content) }; + addToConversation([msg]); + updateStore('messageCountPerSender', receiver.pubkey, 'latest_at', message.created_at); + } + + return success; + } + + const addNewSender = (user: PrimalUser) => { + if (!store.senders[user.pubkey]) { + updateStore('senders', () => ({ [user.pubkey]: {...user }})); + updateStore('messageCountPerSender', user.pubkey, () => ({ cnt: 0 })); + } + + selectSender(user.npub); + }; + + const addSender = (user: PrimalUser) => { + const isFollowing = account?.following.includes(user.pubkey); + + if (isFollowing && store.senderRelation === 'follows' || + !isFollowing && store.senderRelation === 'other' + ) { + addNewSender(user); + return; + } + + updateStore('addSender', () => ({ ...user })); + + changeSenderRelation(isFollowing ? 'follows' : 'other'); + } + + const addUserReference = (user: PrimalUser) => { + updateStore('referecedUsers', () => ({ [user.pubkey]: {...user} })); + }; + + +// SOCKET HANDLERS ------------------------------ + + const onMessage = (event: MessageEvent) => { + const message: NostrEvent | NostrEOSE = JSON.parse(event.data); + + const [type, subId, content] = message; + + if (subId === subidMsgCount) { + if (content?.kind === Kind.MessageStats) { + const count = parseInt(content.cnt); + + if (count !== store.messageCount) { + updateStore('messageCount', () => count); + } + + } + } + + if (subId === subidMsgCountPerSender) { + if (type === 'EVENT') { + if (content?.kind === Kind.MesagePerSenderStats) { + const senderCount = JSON.parse(content.content); + + updateStore('messageCountPerSender', () => ({ ...senderCount })); + updateMessageTimings(); + } + + if (content?.kind === Kind.Metadata) { + if (store.senders[content.pubkey]) { + return; + } + + const isFollowing = account?.following.includes(content.pubkey); + + if (isFollowing && store.senderRelation !== 'follows' || + !isFollowing && store.senderRelation !== 'other' + ) { + return; + } + + const user = convertToUser(content); + + updateStore('senders', () => ({ [user.pubkey]: { ...user } })); + } + } + + if (type === 'EOSE') { + if (store.addSender !== undefined) { + const key = store.addSender.pubkey; + const user = { ...store.addSender } + + updateStore('senders', () => ({ [key]: user })); + updateStore('messageCountPerSender', user.pubkey, () => ({ cnt: 0 })); + selectSender(store.addSender.pubkey); + updateStore('addSender', () => undefined); + return; + } + + const senderIds = Object.keys(store.senders); + if (!store.selectedSender) { + selectSender(senderIds[0]); + } + // !store.selectedSender && updateStore('selectedSender', () => ({ ...store.senders[senderIds[0]] })); + } + } + + if (subId === subidCoversation || subId === subidCoversationNextPage) { + if (type === 'EVENT') { + if (content?.kind === Kind.EncryptedDirectMessage) { + updateStore('encryptedMessages', (conv) => [ ...conv, {...content}]); + } + } + + if (type === 'EOSE') { + if (subId === subidCoversation) { + decryptMessages(generateConversation); + return; + } + + if (subId === subidCoversationNextPage) { + decryptMessages(prependToConversation); + return; + } + } + } + + if (subId === subidNewMsg) { + if (type === 'EVENT') { + if (content?.kind === Kind.EncryptedDirectMessage) { + updateStore('encryptedMessages', (conv) => [ ...conv, {...content}]); + } + } + + if (type === 'EOSE') { + decryptMessages((msgs) => addToConversation(msgs, true)); + } + } + + if (subId === subidUserRef) { + if (type === 'EVENT') { + if (content?.kind === Kind.Metadata) { + const user = content as NostrUserContent; + + updateStore('referencePage', 'users', + (usrs) => ({ ...usrs, [user.pubkey]: { ...user } }) + ); + } + } + + if (type === 'EOSE') { + updateRefUsers(); + } + } + + if (subId === subidNoteRef) { + if (type === 'EVENT') { + if (content?.kind === Kind.Metadata) { + const user = content as NostrUserContent; + + updateStore('referencePage', 'users', + (usrs) => ({ ...usrs, [user.pubkey]: { ...user } }) + ); + } + + if ([Kind.Text, Kind.Repost].includes(content.kind)) { + const message = content as NostrNoteContent; + + updateStore('referencePage', 'messages', + (msgs) => [ ...msgs, { ...message }] + ); + + return; + } + + if (content.kind === Kind.NoteStats) { + const statistic = content as NostrStatsContent; + const stat = JSON.parse(statistic.content); + + updateStore('referencePage', 'postStats', + (stats) => ({ ...stats, [stat.event_id]: { ...stat } }) + ); + return; + } + + if (content.kind === Kind.Mentions) { + const mentionContent = content as NostrMentionContent; + const mention = JSON.parse(mentionContent.content); + + updateStore('referencePage', 'mentions', + (mentions) => ({ ...mentions, [mention.id]: { ...mention } }) + ); + return; + } + + if (content.kind === Kind.NoteActions) { + const noteActionContent = content as NostrNoteActionsContent; + const noteActions = JSON.parse(noteActionContent.content) as NoteActions; + + updateStore('referencePage', 'noteActions', + (actions) => ({ ...actions, [noteActions.event_id]: { ...noteActions } }) + ); + return; + } + } + + if (type === 'EOSE') { + updateRefNotes(); + updateRefUsers(); + } + } + }; + + const onSocketClose = (closeEvent: CloseEvent) => { + const webSocket = closeEvent.target as WebSocket; + + removeSocketListeners( + webSocket, + { message: onMessage, close: onSocketClose }, + ); + }; + +// EFFECTS -------------------------------------- + + createEffect(() => { + if (isConnected() && account?.isKeyLookupDone && account?.hasPublicKey()) { + subToMessagesStats(); + } + }); + + createEffect(() => { + if (isConnected()) { + refreshSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + } + }); + + onCleanup(() => { + removeSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + }); + + let conversationRefreshInterval = 0; + + const updateMessageTimings = () => { + updateStore('now', () => Math.floor((new Date()).getTime() / 1000)); + }; + + // When a sender is selected, get the first page of the conversation + createEffect(() => { + if (store.selectedSender) { + clearInterval(conversationRefreshInterval); + + updateStore('encryptedMessages', () => []); + updateStore('conversation', () => []); + updateStore('messages', () => []); + getConversationWithSender(store.selectedSender); + + conversationRefreshInterval = setInterval(() => { + updateMessageTimings(); + }, 60_000); + } + }); + + // when the total number of messages increases, check for new messages + createEffect(() => { + if ( + account?.hasPublicKey() && + store.selectedSender && + store.messageCountPerSender[store.selectedSender?.pubkey] && + store.messageCountPerSender[store.selectedSender.pubkey].cnt > 0 + ) { + + updateStore('encryptedMessages', () => []); + + let time = Math.floor((new Date()).getTime() / 1000); + + const lastThread = store.conversation[store.conversation.length - 1]; + + if (lastThread) { + const lastMessage = lastThread.messages[lastThread.messages.length - 1]; + + if (lastMessage) { + time = lastMessage.created_at + } + } + + getNewMessages( + // @ts-ignore + account?.publicKey, + store.selectedSender.pubkey, + subidNewMsg, + time, + ); + } + }); + + +// STORES --------------------------------------- + + + const [store, updateStore] = createStore({ + ...initialData, + actions: { + getMessagesPerSender, + selectSender, + addToConversation, + resetConversationLoaded, + sendMessage, + changeSenderRelation, + resetAllMessages, + addSender, + getNextConversationPage, + addUserReference, + }, + }); + +// RENDER --------------------------------------- + + return ( + + {props.children} + + ); +} + +export const useMessagesContext = () => useContext(MessagesContext); diff --git a/src/contexts/NotificationsContext.tsx b/src/contexts/NotificationsContext.tsx new file mode 100644 index 0000000..9d3a9fc --- /dev/null +++ b/src/contexts/NotificationsContext.tsx @@ -0,0 +1,131 @@ +import { createStore } from "solid-js/store"; +import { Kind } from "../constants"; +import { + createContext, + createEffect, + onCleanup, + useContext +} from "solid-js"; +import { + isConnected, + refreshSocketListeners, + removeSocketListeners, + socket +} from "../sockets"; +import { + ContextChildren, + NostrEOSE, + NostrEvent, +} from "../types/primal"; +import { APP_ID } from "../App"; +import { subscribeToNotificationStats } from "../lib/notifications"; +import { useAccountContext } from "./AccountContext"; + +export type NotificationsContextStore = { + notificationCount: number, + actions: { + } +} + +export const initialData = { + notificationCount: 0, +}; + + +export const NotificationsContext = createContext(); + +export const NotificationsProvider = (props: { children: ContextChildren }) => { + + const account = useAccountContext(); + + const subid = `notif_stats_${APP_ID}`; + +// ACTIONS -------------------------------------- + + const subToNotificationStats = () => { + if (!account?.hasPublicKey()) { + return; + } + + // @ts-ignore + subscribeToNotificationStats(account?.publicKey, subid); + } + +// SOCKET HANDLERS ------------------------------ + + const onMessage = (event: MessageEvent) => { + const message: NostrEvent | NostrEOSE = JSON.parse(event.data); + + const [type, subId, content] = message; + + if (subId === subid) { + if (content?.kind === Kind.NotificationStats) { + const sum = Object.keys(content).reduce((acc, key) => { + if (key === 'pubkey' || key == 'kind') { + return acc; + } + + // @ts-ignore + return acc + content[key]; + }, 0); + + if (sum !== store.notificationCount) { + updateStore('notificationCount', () => sum) + } + + } + } + }; + + const onSocketClose = (closeEvent: CloseEvent) => { + const webSocket = closeEvent.target as WebSocket; + + removeSocketListeners( + webSocket, + { message: onMessage, close: onSocketClose }, + ); + }; + +// EFFECTS -------------------------------------- + + createEffect(() => { + if (isConnected() && account?.hasPublicKey()) { + subToNotificationStats(); + } + }); + + createEffect(() => { + if (isConnected()) { + refreshSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + } + }); + + onCleanup(() => { + removeSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + }); + +// STORES --------------------------------------- + + + const [store, updateStore] = createStore({ + ...initialData, + actions: { + }, + }); + +// RENDER --------------------------------------- + + return ( + + {props.children} + + ); +} + +export const useNotificationsContext = () => useContext(NotificationsContext); diff --git a/src/contexts/ProfileContext.tsx b/src/contexts/ProfileContext.tsx new file mode 100644 index 0000000..89915f6 --- /dev/null +++ b/src/contexts/ProfileContext.tsx @@ -0,0 +1,441 @@ +import { nip19 } from "nostr-tools"; +import { createStore } from "solid-js/store"; +import { getEvents, getUserFeed } from "../lib/feed"; +import { convertToNotes, paginationPlan, parseEmptyReposts, sortByRecency, sortByScore } from "../stores/note"; +import { Kind } from "../constants"; +import { + createContext, + createEffect, + onCleanup, + useContext +} from "solid-js"; +import { + isConnected, + refreshSocketListeners, + removeSocketListeners, + socket +} from "../sockets"; +import { + ContextChildren, + FeedPage, + NostrEOSE, + NostrEvent, + NostrEventContent, + NostrMentionContent, + NostrNoteActionsContent, + NostrNoteContent, + NostrStatsContent, + NostrUserContent, + NoteActions, + PrimalNote, + PrimalUser, + VanityProfiles, +} from "../types/primal"; +import { APP_ID } from "../App"; +import { hexToNpub } from "../lib/keys"; +import { + getProfileContactList, + getProfileScoredNotes, + getUserProfileInfo, +} from "../lib/profile"; +import { useAccountContext } from "./AccountContext"; + +export type ProfileContextStore = { + profileKey: string | undefined, + userProfile: PrimalUser | undefined, + userStats: { + follows_count: number, + followers_count: number, + note_count: number, + time_joined: number, + }, + knownProfiles: VanityProfiles, + notes: PrimalNote[], + isFetching: boolean, + page: FeedPage, + reposts: Record | undefined, + lastNote: PrimalNote | undefined, + following: string[], + sidebar: FeedPage & { notes: PrimalNote[] }, + actions: { + saveNotes: (newNotes: PrimalNote[]) => void, + clearNotes: () => void, + fetchNotes: (noteId: string | undefined, until?: number) => void, + fetchNextPage: () => void, + updatePage: (content: NostrEventContent) => void, + savePage: (page: FeedPage) => void, + setProfileKey: (profileKey?: string) => void, + } +} + +export const emptyStats = { + follows_count: 0, + followers_count: 0, + note_count: 0, + time_joined: 0, +}; + +export const initialData = { + profileKey: undefined, + userProfile: undefined, + userStats: { ...emptyStats }, + knownProfiles: { names: {} }, + notes: [], + isFetching: false, + page: { messages: [], users: {}, postStats: {}, mentions: {}, noteActions: {} }, + reposts: {}, + lastNote: undefined, + following: [], + sidebar: { + messages: [], + users: {}, + postStats: {}, + notes: [], + noteActions: {}, + }, +}; + + +export const ProfileContext = createContext(); + +export const ProfileProvider = (props: { children: ContextChildren }) => { + + const account = useAccountContext(); + +// ACTIONS -------------------------------------- + + const saveNotes = (newNotes: PrimalNote[]) => { + updateStore('notes', (notes) => [ ...notes, ...newNotes ]); + updateStore('isFetching', () => false); + }; + + const fetchNotes = (pubkey: string | undefined, until = 0, limit = 20) => { + if (!pubkey) { + return; + } + + updateStore('isFetching', () => true); + updateStore('page', () => ({ messages: [], users: {}, postStats: {} })); + getUserFeed(account?.publicKey, pubkey, `profile_feed_${APP_ID}`, until, limit); + } + + const clearNotes = () => { + updateStore('page', () => ({ messages: [], users: {}, postStats: {}, noteActions: {} })); + updateStore('notes', () => []); + updateStore('reposts', () => undefined); + updateStore('lastNote', () => undefined); + updateStore('sidebar', () => ({ + messages: [], + users: {}, + postStats: {}, + notes: [], + noteActions: {}, + })); + }; + + const fetchNextPage = () => { + const lastNote = store.notes[store.notes.length - 1]; + + if (!lastNote) { + return; + } + + updateStore('lastNote', () => ({ ...lastNote })); + + const criteria = paginationPlan('latest'); + + const noteData: Record = lastNote.repost ? + lastNote.repost.note : + lastNote.post; + + const until = noteData[criteria]; + + if (until > 0 && store.profileKey) { + fetchNotes(store.profileKey, until); + } + }; + + const updatePage = (content: NostrEventContent) => { + if (content.kind === Kind.Metadata) { + const user = content as NostrUserContent; + + updateStore('page', 'users', () => ({ [user.pubkey]: user})); + return; + } + + if ([Kind.Text, Kind.Repost].includes(content.kind)) { + const message = content as NostrNoteContent; + const messageId = nip19.noteEncode(message.id); + + const isLastNote = message.kind === Kind.Text ? + store.lastNote?.post?.noteId === messageId : + store.lastNote?.repost?.note.noteId === messageId; + + if (!isLastNote) { + updateStore('page', 'messages', messages => [ ...messages, message]); + } + + return; + } + + if (content.kind === Kind.NoteStats) { + const statistic = content as NostrStatsContent; + const stat = JSON.parse(statistic.content); + + updateStore('page', 'postStats', () => ({ [stat.event_id]: stat })); + return; + } + + if (content.kind === Kind.Mentions) { + const mentionContent = content as NostrMentionContent; + const mention = JSON.parse(mentionContent.content); + + updateStore('page', 'mentions', () => ({ [mention.id]: mention })); + return; + } + + if (content.kind === Kind.NoteActions) { + const noteActionContent = content as NostrNoteActionsContent; + const noteActions = JSON.parse(noteActionContent.content) as NoteActions; + + updateStore('page', 'noteActions', () => ({ [noteActions.event_id]: noteActions })); + return; + } + }; + + const savePage = (page: FeedPage) => { + const newPosts = sortByRecency(convertToNotes(page)); + + saveNotes(newPosts); + }; + + + const updateSidebar = (content: NostrEventContent) => { + if (content.kind === Kind.Metadata) { + const user = content as NostrUserContent; + + updateStore('sidebar', 'users', () => ({ [user.pubkey]: user }) + ); + return; + } + + if ([Kind.Text, Kind.Repost].includes(content.kind)) { + const message = content as NostrNoteContent; + + if (store.lastNote?.post?.noteId !== nip19.noteEncode(message.id)) { + updateStore('sidebar', 'messages', (msgs) => [ ...msgs, message ]); + } + + return; + } + + if (content.kind === Kind.NoteStats) { + const statistic = content as NostrStatsContent; + const stat = JSON.parse(statistic.content); + + updateStore('sidebar', 'postStats', () => ({ [stat.event_id]: stat })); + return; + } + + if (content.kind === Kind.Mentions) { + const mentionContent = content as NostrMentionContent; + const mention = JSON.parse(mentionContent.content); + + updateStore('page', 'mentions', () => ({ [mention.id]: mention })); + return; + } + + if (content.kind === Kind.NoteActions) { + const noteActionContent = content as NostrNoteActionsContent; + const noteActions = JSON.parse(noteActionContent.content) as NoteActions; + + updateStore('page', 'noteActions', () => ({ [noteActions.event_id]: noteActions })); + return; + } + }; + + const saveSidebar = (page: FeedPage) => { + const newPosts = sortByScore(convertToNotes(page)); + + updateStore('sidebar', 'notes', () => [ ...newPosts ]); + }; + + const setProfileKey = (profileKey?: string) => { + updateStore('profileKey', () => profileKey); + + if (profileKey) { + updateStore('userProfile', () => undefined); + updateStore('userStats', () => ({ ...emptyStats })); + getUserProfileInfo(profileKey, `profile_info_${APP_ID}`); + getProfileContactList(profileKey, `profile_contacts_${APP_ID}`); + getProfileScoredNotes(profileKey, `profile_scored_${APP_ID}`, 10); + } + } + +// SOCKET HANDLERS ------------------------------ + + const onMessage = (event: MessageEvent) => { + const message: NostrEvent | NostrEOSE = JSON.parse(event.data); + + const [type, subId, content] = message; + + if (subId === `profile_feed_${APP_ID}`) { + if (type === 'EOSE') { + const reposts = parseEmptyReposts(store.page); + const ids = Object.keys(reposts); + + if (ids.length === 0) { + savePage(store.page); + return; + } + + updateStore('reposts', () => reposts); + + getEvents(account?.publicKey, ids, `profile_reposts_${APP_ID}`); + return; + } + + if (type === 'EVENT') { + updatePage(content); + return; + } + } + + if (subId === `profile_info_${APP_ID}`) { + + if (content?.kind === Kind.Metadata) { + let user = JSON.parse(content.content); + + if (!user.displayName || typeof user.displayName === 'string' && user.displayName.trim().length === 0) { + user.displayName = user.display_name; + } + user.pubkey = content.pubkey; + user.npub = hexToNpub(content.pubkey); + user.created_at = content.created_at; + + updateStore('userProfile', () => user); + return; + } + + if (content?.kind === Kind.UserStats) { + const stats = JSON.parse(content.content); + + updateStore('userStats', () => ({ ...stats })); + return; + } + } + + if (subId === `profile_reposts_${APP_ID}`) { + if (type === 'EOSE') { + savePage(store.page); + return; + } + + if (type === 'EVENT') { + const repostId = (content as NostrNoteContent).id; + const reposts = store.reposts || {}; + const parent = store.page.messages.find(m => m.id === reposts[repostId]); + + if (parent) { + updateStore('page', 'messages', (msg) => msg.id === parent.id, 'content', () => JSON.stringify(content)); + } + + return; + } + } + + // if (subId === `profile_oldest_${APP_ID}`) { + // if (content?.kind === Kind.OldestEvent) { + // const timestamp = Number.parseInt(content.content); + // if (isNaN(timestamp)) { + // updateStore('oldestNoteDate', () => undefined); + // return; + // } + // updateStore('oldestNoteDate', () => timestamp); + // } + // return; + // } + + if (subId === `profile_contacts_${APP_ID}`) { + if (content && content.kind === Kind.Contacts) { + const tags = content.tags; + let contacts: string[] = []; + + for (let i = 0;i contacts); + } + return; + } + + if (subId === `profile_scored_${APP_ID}`) { + if (type === 'EOSE') { + saveSidebar(store.sidebar); + return; + } + + if (type === 'EVENT') { + updateSidebar(content); + return; + } + } + }; + + const onSocketClose = (closeEvent: CloseEvent) => { + const webSocket = closeEvent.target as WebSocket; + + removeSocketListeners( + webSocket, + { message: onMessage, close: onSocketClose }, + ); + }; + +// EFFECTS -------------------------------------- + + createEffect(() => { + if (isConnected()) { + refreshSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + } + }); + + onCleanup(() => { + removeSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + }); + +// STORES --------------------------------------- + + + const [store, updateStore] = createStore({ + ...initialData, + actions: { + saveNotes, + clearNotes, + fetchNotes, + fetchNextPage, + updatePage, + savePage, + setProfileKey, + }, + }); + +// RENDER --------------------------------------- + + return ( + + {props.children} + + ); +} + +export const useProfileContext = () => useContext(ProfileContext); diff --git a/src/contexts/SearchContext.tsx b/src/contexts/SearchContext.tsx new file mode 100644 index 0000000..c30a04e --- /dev/null +++ b/src/contexts/SearchContext.tsx @@ -0,0 +1,413 @@ +import { createStore } from "solid-js/store"; +import { + createContext, + JSX, + useContext +} from "solid-js"; +import { + FeedPage, + NostrEventContent, + NostrMentionContent, + NostrNoteActionsContent, + NostrNoteContent, + NostrStatsContent, + NostrUserContent, + NoteActions, + PrimalNote, + PrimalUser, +} from '../types/primal'; +import { Kind } from "../constants"; +import { APP_ID } from "../App"; +import { getUserProfiles } from "../lib/profile"; +import { searchContent, searchUsers } from "../lib/search"; +import { convertToUser } from "../stores/profile"; +import { sortByRecency, convertToNotes } from "../stores/note"; +import { subscribeTo } from "../sockets"; +import { nip19 } from "nostr-tools"; + +const recomendedUsers = [ + '82341f882b6eabcd2ba7f1ef90aad961cf074af15b9ef44a09f9d2a8fbfbe6a2', // jack + 'bf2376e17ba4ec269d10fcc996a4746b451152be9031fa48e74553dde5526bce', // carla + 'c48e29f04b482cc01ca1f9ef8c86ef8318c059e0e9353235162f080f26e14c11', // walker + '85080d3bad70ccdcd7f74c29a44f55bb85cbcd3dd0cbb957da1d215bdb931204', // preston + 'eab0e756d32b80bcd464f3d844b8040303075a13eabc3599a762c9ac7ab91f4f', // lyn + '04c915daefee38317fa734444acee390a8269fe5810b2241e5e6dd343dfbecc9', // odell + '472f440f29ef996e92a186b8d320ff180c855903882e59d50de1b8bd5669301e', // marty + 'e88a691e98d9987c964521dff60025f60700378a4879180dcbbb4a5027850411', // nvk + '91c9a5e1a9744114c6fe2d61ae4de82629eaaa0fb52f48288093c7e7e036f832', // rockstar + 'fa984bd7dbb282f07e16e7ae87b26a2a7b9b90b7246a44771f0cf5ae58018f52', // pablo +]; + +export type SearchContextStore = { + contentQuery: string, + users: PrimalUser[], + scores: Record, + contentUsers: PrimalUser[], + contentScores: Record, + notes: PrimalNote[], + isFetchingUsers: boolean, + isFetchingContent: boolean, + page: FeedPage, + reposts: Record | undefined, + mentionedNotes: Record, + actions: { + findUsers: (query: string, pubkey?: string) => void, + findUserByNupub: (npub: string) => void, + findContentUsers: (query: string, pubkey?: string) => void, + findContent: (query: string) => void, + setContentQuery: (query: string) => void, + getRecomendedUsers: () => void, + }, +} + +const initialData = { + contentQuery: '', + users: [], + scores: {}, + contentUsers: [], + contentScores: {}, + notes: [], + isFetchingUsers: false, + isFetchingContent: false, + page: { messages: [], users: {}, postStats: {}, mentions: {}, noteActions: {} }, + reposts: {}, + mentionedNotes: {}, +}; + +export const SearchContext = createContext(); + +export function SearchProvider(props: { children: number | boolean | Node | JSX.ArrayElement | JSX.FunctionElement | (string & {}) | null | undefined; }) { + +// ACTIONS -------------------------------------- + + const findUserByNupub = (npub: string) => { + const subId = `find_npub_${APP_ID}`; + + let decoded: nip19.DecodeResult | undefined; + + try { + decoded = nip19.decode(npub); + } catch (e) { + findUsers(npub); + return; + } + + if (!decoded) { + findUsers(npub); + return; + } + + const hex = typeof decoded.data === 'string' ? + decoded.data : + (decoded.data as nip19.ProfilePointer).pubkey; + + let users: PrimalUser[] = []; + + const unsub = subscribeTo(subId, (type, _, content) => { + if (type === 'EVENT') { + if (!content) { + return; + } + + if (content.kind === Kind.Metadata) { + const user = content as NostrUserContent; + + users.push(convertToUser(user)); + return; + } + + if (content.kind === Kind.UserScore) { + const scores = JSON.parse(content.content); + + updateStore('scores', () => ({ ...scores })); + return; + } + } + + if (type === 'EOSE') { + + if (users.length > 0) { + updateStore('users', () => [users[0]]); + } + + updateStore('isFetchingUsers', () => false); + + unsub(); + return; + } + }); + + getUserProfiles([hex], subId); + }; + + const getRecomendedUsers = () => { + const subid = `recomended_users_${APP_ID}`; + + let users: PrimalUser[] = []; + + const unsub = subscribeTo(subid, (type, _, content) => { + if (type === 'EVENT') { + if (!content) { + return; + } + + if (content.kind === Kind.Metadata) { + const user = content as NostrUserContent; + + users.push(convertToUser(user)); + return; + } + + if (content.kind === Kind.UserScore) { + const scores = JSON.parse(content.content); + + updateStore('scores', () => ({ ...scores })); + return; + } + } + + if (type === 'EOSE') { + + let sorted: PrimalUser[] = []; + + users.forEach((user) => { + const index = recomendedUsers.indexOf(user.pubkey); + sorted[index] = { ...user }; + }); + + updateStore('users', () => sorted); + updateStore('isFetchingUsers', () => false); + + unsub(); + return; + } + }); + + + updateStore('isFetchingUsers', () => true); + getUserProfiles(recomendedUsers, subid); + + }; + + const findUsers = (query: string, publicKey?: string) => { + const subid = `search_users_${APP_ID}`; + + let users: PrimalUser[] = []; + + const unsub = subscribeTo(subid, (type, _, content) => { + if (type === 'EVENT') { + if (!content) { + return; + } + + if (content.kind === Kind.Metadata) { + const user = content as NostrUserContent; + + users.push(convertToUser(user)); + return; + } + + if (content.kind === Kind.UserScore) { + const scores = JSON.parse(content.content); + + updateStore('scores', () => ({ ...scores })); + return; + } + } + + if (type === 'EOSE') { + const sorted = users.sort((a, b) => { + const aScore = store.scores[a.pubkey]; + const bScore = store.scores[b.pubkey]; + + return bScore - aScore; + }); + + updateStore('users', () => sorted.slice(0, 10)); + updateStore('isFetchingUsers', () => false); + + unsub(); + return; + } + + }); + + const pubkey = query.length > 0 ? undefined : publicKey; + + updateStore('isFetchingUsers', () => true); + searchUsers(pubkey, subid, query); + } + + const findContentUsers = (query: string, publicKey?: string) => { + const subid = `search_users_c_${APP_ID}`; + + let users: PrimalUser[] = []; + + const unsub = subscribeTo(subid, (type, _, content) => { + if (type === 'EVENT') { + if (!content) { + return; + } + + if (content.kind === Kind.Metadata) { + const user = content as NostrUserContent; + + users.push(convertToUser(user)); + return; + } + + if (content.kind === Kind.UserScore) { + const scores = JSON.parse(content.content); + + updateStore('contentScores', () => ({ ...scores })); + return; + } + } + + if (type === 'EOSE') { + const sorted = users.sort((a, b) => { + const aScore = store.scores[a.pubkey]; + const bScore = store.scores[b.pubkey]; + + return bScore - aScore; + }); + + updateStore('contentUsers', () => sorted.slice(0, 10)); + + unsub(); + return; + } + + }); + + const pubkey = query.length > 0 ? undefined : publicKey; + + updateStore('isFetchingUsers', () => true); + searchUsers(pubkey, subid, query); + } + + const saveNotes = (newNotes: PrimalNote[]) => { + updateStore('notes', () => [ ...newNotes ]); + updateStore('isFetchingContent', () => false); + }; + + + const updatePage = (content: NostrEventContent) => { + if (content.kind === Kind.Metadata) { + const user = content as NostrUserContent; + + updateStore('page', 'users', + (usrs) => ({ ...usrs, [user.pubkey]: { ...user } }) + ); + return; + } + + if ([Kind.Text, Kind.Repost].includes(content.kind)) { + const message = content as NostrNoteContent; + + updateStore('page', 'messages', + (msgs) => [ ...msgs, { ...message }] + ); + + return; + } + + if (content.kind === Kind.NoteStats) { + const statistic = content as NostrStatsContent; + const stat = JSON.parse(statistic.content); + + updateStore('page', 'postStats', + (stats) => ({ ...stats, [stat.event_id]: { ...stat } }) + ); + return; + } + + if (content.kind === Kind.Mentions) { + const mentionContent = content as NostrMentionContent; + const mention = JSON.parse(mentionContent.content); + + updateStore('page', 'mentions', + (mentions) => ({ ...mentions, [mention.id]: { ...mention } }) + ); + return; + } + + if (content.kind === Kind.NoteActions) { + const noteActionContent = content as NostrNoteActionsContent; + const noteActions = JSON.parse(noteActionContent.content) as NoteActions; + + updateStore('page', 'noteActions', + (actions) => ({ ...actions, [noteActions.event_id]: { ...noteActions } }) + ); + return; + } + }; + + const savePage = (page: FeedPage) => { + const newPosts = sortByRecency(convertToNotes(page)); + + saveNotes(newPosts); + }; + + const findContent = (query: string) => { + const subid = `search_content_${APP_ID}`; + + const unsub = subscribeTo(subid, (type, _, content) => { + + if (type === 'EOSE') { + savePage(store.page); + unsub(); + return; + } + + if (!content) { + return; + } + + + if (type === 'EVENT') { + updatePage(content); + return; + } + + }); + + updateStore('isFetchingContent', () => true); + updateStore('notes', () => []); + updateStore('page', { messages: [], users: {}, postStats: {}, mentions: {}, noteActions: {} }) + searchContent(subid, query); + } + + const setContentQuery = (query: string) => { + updateStore('contentQuery', () => query); + }; + + + +// EFFECTS -------------------------------------- + +// SOCKET HANDLERS ------------------------------ + + +// STORES --------------------------------------- + +const [store, updateStore] = createStore({ + ...initialData, + actions: { + findUsers, + findUserByNupub, + findContent, + findContentUsers, + setContentQuery, + getRecomendedUsers, + }, +}); + + return ( + + {props.children} + + ); +} + +export function useSearchContext() { return useContext(SearchContext); } diff --git a/src/contexts/SettingsContext.tsx b/src/contexts/SettingsContext.tsx new file mode 100644 index 0000000..3ac3c3d --- /dev/null +++ b/src/contexts/SettingsContext.tsx @@ -0,0 +1,444 @@ +import { createStore } from "solid-js/store"; +import { useToastContext } from "../components/Toaster/Toaster"; +import { defaultFeeds, noKey, defaultNotificationSettings, themes, trendingFeed } from "../constants"; +import { + createContext, + createEffect, + onCleanup, + onMount, + useContext +} from "solid-js"; +import { + isConnected, + refreshSocketListeners, + removeSocketListeners, + socket, + subscribeTo +} from "../sockets"; +import { + ContextChildren, + PrimalFeed, + PrimalTheme, +} from "../types/primal"; +import { + initAvailableFeeds, + removeFromAvailableFeeds, + replaceAvailableFeeds, + updateAvailableFeeds, + updateAvailableFeedsTop +} from "../lib/availableFeeds"; +import { useAccountContext } from "./AccountContext"; +import { saveTheme } from "../lib/localStore"; +import { getDefaultSettings, getSettings, sendSettings } from "../lib/settings"; +import { APP_ID } from "../App"; +import { useIntl } from "@cookbook/solid-intl"; +import { hexToNpub } from "../lib/keys"; + +export type SettingsContextStore = { + locale: string, + theme: string, + themes: PrimalTheme[], + availableFeeds: PrimalFeed[], + defaultFeed: PrimalFeed, + defaultZapAmount: number, + availableZapOptions: number[], + notificationSettings: Record, + actions: { + setTheme: (theme: PrimalTheme | null) => void, + addAvailableFeed: (feed: PrimalFeed, addToTop?: boolean) => void, + removeAvailableFeed: (feed: PrimalFeed) => void, + setAvailableFeeds: (feedList: PrimalFeed[]) => void, + moveAvailableFeed: (fromIndex: number, toIndex: number) => void, + renameAvailableFeed: (feed: PrimalFeed, newName: string) => void, + saveSettings: () => void, + loadSettings: (pubkey: string) => void, + setDefaultZapAmount: (amount: number) => void, + setZapOptions: (amount:number, index: number) => void, + updateNotificationSettings: (key: string, value: boolean, temp?: boolean) => void, + } +} + +export const initialData = { + locale: 'en-us', + theme: 'sunset', + themes, + availableFeeds: [], + defaultFeed: defaultFeeds[0], + defaultZapAmount: 10, + availableZapOptions: [ + 21, + 420, + 10_000, + 69_420, + 100_000, + 1_000_000, + ], + notificationSettings: { ...defaultNotificationSettings }, +}; + + +export const SettingsContext = createContext(); + +export const SettingsProvider = (props: { children: ContextChildren }) => { + + const toaster = useToastContext(); + const account = useAccountContext(); + const intl = useIntl(); + +// ACTIONS -------------------------------------- + + const setDefaultZapAmount = (amount: number, temp?: boolean) => { + updateStore('defaultZapAmount', () => amount); + !temp && saveSettings(); + }; + + const setZapOptions = (amount: number, index: number, temp?: boolean) => { + updateStore('availableZapOptions', index, () => amount); + !temp && saveSettings(); + }; + + const setTheme = (theme: PrimalTheme | null, temp?: boolean) => { + if (!theme) { + return; + } + + saveTheme(account?.publicKey, theme.name); + updateStore('theme', () => theme.name); + !temp && saveSettings(); + } + + const setThemeByName = (name: string | null, temp?: boolean) => { + if (!name) { + return; + } + + const availableTheme = store.themes.find(t => t.name === name); + availableTheme && setTheme(availableTheme, temp); + } + + const addAvailableFeed = (feed: PrimalFeed, addToTop = false, temp?: boolean) => { + if (!feed) { + return; + } + if (account?.hasPublicKey()) { + const add = addToTop ? updateAvailableFeedsTop : updateAvailableFeeds; + + updateStore('availableFeeds', (feeds) => add(account?.publicKey, feed, feeds)); + !temp && saveSettings(); + } + }; + + const removeAvailableFeed = (feed: PrimalFeed, temp?: boolean) => { + if (!feed) { + return; + } + + if (account?.hasPublicKey()) { + updateStore('availableFeeds', + (feeds) => removeFromAvailableFeeds(account?.publicKey, feed, feeds), + ); + + !temp && saveSettings(); + toaster?.sendSuccess(`"${feed.name}" has been removed from your home page`); + } + }; + + const setAvailableFeeds = (feedList: PrimalFeed[], temp?: boolean) => { + if (account?.hasPublicKey()) { + updateStore('availableFeeds', + () => replaceAvailableFeeds(account?.publicKey, feedList), + ); + !temp && saveSettings(); + } + }; + + const moveAvailableFeed = (fromIndex: number, toIndex: number) => { + + let list = [...store.availableFeeds]; + + list.splice(toIndex, 0, list.splice(fromIndex, 1)[0]); + + setAvailableFeeds(list); + + }; + + const renameAvailableFeed = (feed: PrimalFeed, newName: string) => { + const list = store.availableFeeds.map(af => { + return af.hex === feed.hex ? { ...af, name: newName } : { ...af }; + }); + setAvailableFeeds(list); + }; + + const updateNotificationSettings = (key: string, value: boolean, temp?: boolean) => { + updateStore('notificationSettings', () => ({ [key]: value })); + + !temp && saveSettings(); + }; + + const saveSettings = () => { + const settings = { + theme: store.theme, + feeds: store.availableFeeds, + defaultZapAmount: store.defaultZapAmount, + zapOptions: store.availableZapOptions, + notifications: store.notificationSettings, + }; + + const subid = `save_settings_${APP_ID}`; + + const unsub = subscribeTo(subid, async (type, subId, content) => { + if (type === 'NOTICE') { + toaster?.sendWarning(intl.formatMessage({ + id: 'settings.saveFail', + defaultMessage: 'Failed to save settings', + description: 'Toast message after settings have failed to be saved on the server', + })); + } + + unsub(); + return; + }); + + sendSettings(settings, subid); + } + + const loadDefaults = () => { + + const subid = `load_defaults_${APP_ID}`; + + const unsub = subscribeTo(subid, async (type, subId, content) => { + + if (type === 'EVENT' && content?.content) { + try { + const settings = JSON.parse(content?.content); + + const feeds = settings.feeds as PrimalFeed[]; + const notificationSettings = settings.notifications as Record; + + // const availableTopics = store.availableFeeds.map(f => f.hex); + + // const updatedFeeds = feeds.reduce((acc, feed) => { + // return availableTopics.includes(feed.hex) ? + // acc : + // [ ...acc, feed ]; + // }, store.availableFeeds) + + updateStore('availableFeeds', + () => replaceAvailableFeeds(account?.publicKey, feeds), + ); + + updateStore('defaultFeed', () => store.availableFeeds[0]); + + updateStore('notificationSettings', () => ({ ...notificationSettings } || { ...defaultNotificationSettings })); + + const defaultZaps = settings.defaultZapAmount || 10; + + const zapOptions = settings.zapOptions || [ + 21, + 420, + 10_000, + 69_420, + 100_000, + 1_000_000, + ]; + + updateStore('defaultZapAmount', () => defaultZaps); + updateStore('availableZapOptions', () => zapOptions); + } + catch (e) { + console.log('Error parsing settings response: ', e); + } + } + + if (type === 'NOTICE') { + toaster?.sendWarning(intl.formatMessage({ + id: 'settings.loadFail', + defaultMessage: 'Failed to load settings. Will be using local settings.', + description: 'Toast message after settings have failed to be loaded from the server', + })); + } + + unsub(); + return; + }); + + getDefaultSettings(subid) + }; + + const loadSettings = (pubkey: string | undefined) => { + if (!pubkey || pubkey === noKey) { + return; + } + + const subid = `load_settings_${APP_ID}`; + + const unsub = subscribeTo(subid, async (type, subId, content) => { + + if (type === 'EVENT' && content?.content) { + try { + const { theme, feeds, defaultZapAmount, zapOptions, notifications } = JSON.parse(content?.content); + + theme && setThemeByName(theme, true); + feeds && setAvailableFeeds(feeds, true); + defaultZapAmount && setDefaultZapAmount(defaultZapAmount, true); + zapOptions && updateStore('availableZapOptions', () => zapOptions); + + if (notifications) { + updateStore('notificationSettings', () => ({ ...notifications })); + } + else { + updateStore('notificationSettings', () => ({ ...defaultNotificationSettings})); + } + } + catch (e) { + console.log('Error parsing settings response: ', e); + } + } + + if (type === 'NOTICE') { + toaster?.sendWarning(intl.formatMessage({ + id: 'settings.loadFail', + defaultMessage: 'Failed to load settings. Will be using local settings.', + description: 'Toast message after settings have failed to be loaded from the server', + })); + } + + updateStore('defaultFeed', () => store.availableFeeds[0]); + + unsub(); + return; + }); + + pubkey && getSettings(pubkey, subid); + } + +// SOCKET HANDLERS ------------------------------ + + const onMessage = (event: MessageEvent) => { + // const message: NostrEvent | NostrEOSE = JSON.parse(event.data); + + // const [type, subId, content] = message; + }; + + const onSocketClose = (closeEvent: CloseEvent) => { + const webSocket = closeEvent.target as WebSocket; + + removeSocketListeners( + webSocket, + { message: onMessage, close: onSocketClose }, + ); + }; + + +// EFFECTS -------------------------------------- + + onMount(() => { + // Set global theme, this is done to avoid changing the theme + // when waiting for pubkey (like when reloading a page). + const storedTheme = localStorage.getItem('theme'); + setThemeByName(storedTheme, true); + }); + + + // This is here as to not trigger the effect + // TODO Solve this. + const feedLabel = intl.formatMessage({ + id: 'feeds.latestFollowing', + defaultMessage: 'Latest, following', + description: 'Label for the `latest;following` (active user\'s) feed', + }); + + + // Initial setup for a user with a public key + createEffect(() => { + if (!account?.hasPublicKey() && account?.isKeyLookupDone) { + loadDefaults(); + return; + } + + const pubkey = account?.publicKey; + + const initFeeds = initAvailableFeeds(pubkey); + + if (initFeeds && initFeeds.length > 0) { + updateStore('defaultFeed', () => initFeeds[0]); + updateStore('availableFeeds', () => replaceAvailableFeeds(pubkey, initFeeds)); + } + + + const feed = { + name: feedLabel, + hex: pubkey, + npub: hexToNpub(pubkey), + }; + + // Add trendingFeed if it's missing + // @ts-ignore + if (initFeeds && !initFeeds.find((f) => f.hex === trendingFeed.hex)) { + addAvailableFeed(trendingFeed, true, true); + } + + // Add active user's feed if it's missing + // @ts-ignore + if (initFeeds && !initFeeds.find(f => f.hex === feed.hex)) { + addAvailableFeed(feed, true, true); + } + + + setTimeout(() => { + loadSettings(pubkey); + }, 100); + }); + + createEffect(() => { + const html: HTMLElement | null = document.querySelector('html'); + localStorage.setItem('theme', store.theme); + html?.setAttribute('data-theme', store.theme); + }); + + createEffect(() => { + if (isConnected()) { + refreshSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + } + }); + + onCleanup(() => { + removeSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + }); + +// STORES --------------------------------------- + + + const [store, updateStore] = createStore({ + ...initialData, + actions: { + setTheme, + addAvailableFeed, + removeAvailableFeed, + setAvailableFeeds, + moveAvailableFeed, + renameAvailableFeed, + saveSettings, + loadSettings, + setDefaultZapAmount, + setZapOptions, + updateNotificationSettings, + }, + }); + +// RENDER --------------------------------------- + + return ( + + {props.children} + + ); +} + +export const useSettingsContext = () => useContext(SettingsContext); diff --git a/src/contexts/ThreadContext.tsx b/src/contexts/ThreadContext.tsx new file mode 100644 index 0000000..3ef8069 --- /dev/null +++ b/src/contexts/ThreadContext.tsx @@ -0,0 +1,317 @@ +import { nip19 } from "nostr-tools"; +import { createStore } from "solid-js/store"; +import { getEvents, getThread } from "../lib/feed"; +import { + convertToNotes, + parseEmptyReposts, + sortByRecency, +} from "../stores/note"; +import { convertToUser } from "../stores/profile"; +import { Kind } from "../constants"; +import { + createContext, + createEffect, + onCleanup, + useContext +} from "solid-js"; +import { + isConnected, + refreshSocketListeners, + removeSocketListeners, + socket +} from "../sockets"; +import { + ContextChildren, + FeedPage, + NostrEOSE, + NostrEvent, + NostrEventContent, + NostrMentionContent, + NostrNoteActionsContent, + NostrNoteContent, + NostrStatsContent, + NostrUserContent, + NoteActions, + PrimalNote, + PrimalUser, +} from "../types/primal"; +import { APP_ID } from "../App"; +import { useAccountContext } from "./AccountContext"; + +export type ThreadContextStore = { + primaryNote: PrimalNote | undefined, + noteId: string; + notes: PrimalNote[], + users: PrimalUser[], + isFetching: boolean, + page: FeedPage, + reposts: Record | undefined, + lastNote: PrimalNote | undefined, + actions: { + saveNotes: (newNotes: PrimalNote[]) => void, + clearNotes: () => void, + fetchNotes: (noteId: string, until?: number) => void, + fetchNextPage: () => void, + updatePage: (content: NostrEventContent) => void, + savePage: (page: FeedPage) => void, + setPrimaryNote: (context: PrimalNote | undefined) => void, + } +} + +export const initialData = { + primaryNote: undefined, + noteId: '', + parentNotes: [], + notes: [], + users: [], + replyNotes: [], + isFetching: false, + page: { + messages: [], + users: {}, + postStats: {}, + mentions: {}, + noteActions: {}, + }, + reposts: {}, + lastNote: undefined, +}; + + +export const ThreadContext = createContext(); + +export const ThreadProvider = (props: { children: ContextChildren }) => { + + const account = useAccountContext(); + +// ACTIONS -------------------------------------- + + const saveNotes = (newNotes: PrimalNote[]) => { + updateStore('notes', (notes) => [ ...notes, ...newNotes ]); + updateStore('isFetching', () => false); + }; + + const fetchNotes = (noteId: string, until = 0, limit = 100) => { + clearNotes(); + updateStore('noteId', noteId) + getThread(account?.publicKey, noteId, `thread_${APP_ID}`); + updateStore('isFetching', () => true); + } + + const clearNotes = () => { + updateStore('page', () => ({ messages: [], users: {}, postStats: {}, noteActions: {} })); + updateStore('notes', () => []); + updateStore('reposts', () => undefined); + updateStore('lastNote', () => undefined); + }; + + const fetchNextPage = () => { + const lastNote = store.notes[store.notes.length - 1]; + + if (!lastNote) { + return; + } + + updateStore('lastNote', () => ({ ...lastNote })); + + // Disable pagination for thread feeds + const until = 0; //lastNote.post?.created_at || 0; + + if (until > 0) { + fetchNotes(store.noteId); + } + }; + + const updatePage = (content: NostrEventContent) => { + if (content.kind === Kind.Metadata) { + const user = content as NostrUserContent; + + updateStore('page', 'users', + (usrs) => ({ ...usrs, [user.pubkey]: { ...user } }) + ); + return; + } + + if ([Kind.Text, Kind.Repost].includes(content.kind)) { + const message = content as NostrNoteContent; + + if (store.lastNote?.post?.noteId !== nip19.noteEncode(message.id)) { + updateStore('page', 'messages', + (msgs) => [ ...msgs, { ...message }] + ); + } + + return; + } + + if (content.kind === Kind.NoteStats) { + const statistic = content as NostrStatsContent; + const stat = JSON.parse(statistic.content); + + updateStore('page', 'postStats', + (stats) => ({ ...stats, [stat.event_id]: { ...stat } }) + ); + return; + } + + if (content.kind === Kind.Mentions) { + const mentionContent = content as NostrMentionContent; + const mention = JSON.parse(mentionContent.content); + + updateStore('page', 'mentions', + (mentions) => ({ ...mentions, [mention.id]: { ...mention } }) + ); + return; + } + + if (content.kind === Kind.NoteActions) { + const noteActionContent = content as NostrNoteActionsContent; + const noteActions = JSON.parse(noteActionContent.content) as NoteActions; + + updateStore('page', 'noteActions', + (actions) => ({ ...actions, [noteActions.event_id]: { ...noteActions } }) + ); + return; + } + }; + + const savePage = (page: FeedPage) => { + const newPosts = sortByRecency(convertToNotes(page)); + const users = Object.values(page.users).map(convertToUser); + + updateStore('users', () => [ ...users ]); + saveNotes(newPosts); + }; + + const setPrimaryNote = (context: PrimalNote | undefined) => { + updateStore('primaryNote', () => ({ ...context })); + }; + +// SOCKET HANDLERS ------------------------------ + + const onMessage = (event: MessageEvent) => { + const message: NostrEvent | NostrEOSE = JSON.parse(event.data); + + const [type, subId, content] = message; + + if (subId === `thread_${APP_ID}`) { + if (type === 'EOSE') { + const reposts = parseEmptyReposts(store.page); + const ids = Object.keys(reposts); + + if (ids.length === 0) { + savePage(store.page); + return; + } + + updateStore('reposts', () => reposts); + + getEvents(account?.publicKey, ids, `thread_reposts_${APP_ID}`); + + return; + } + + if (type === 'EVENT') { + updatePage(content); + return; + } + } + + if (subId === `thread_reposts_${APP_ID}`) { + if (type === 'EOSE') { + savePage(store.page); + return; + } + + if (type === 'EVENT') { + const repostId = (content as NostrNoteContent).id; + const reposts = store.reposts || {}; + const parent = store.page.messages.find(m => m.id === reposts[repostId]); + + if (parent) { + updateStore('page', 'messages', (msg) => msg.id === parent.id, 'content', () => JSON.stringify(content)); + } + + return; + } + } + }; + + const onSocketClose = (closeEvent: CloseEvent) => { + const webSocket = closeEvent.target as WebSocket; + + removeSocketListeners( + webSocket, + { message: onMessage, close: onSocketClose }, + ); + }; + +// EFFECTS -------------------------------------- + + createEffect(() => { + if (isConnected()) { + refreshSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + } + }); + + onCleanup(() => { + removeSocketListeners( + socket(), + { message: onMessage, close: onSocketClose }, + ); + }); + +// STORES --------------------------------------- + + const primaryNote: () => PrimalNote | undefined = () => + store.notes.find(n => n.post.id === store.noteId); + + const parentNotes: () => PrimalNote[] = () => { + const note = primaryNote(); + + if (!note) { + return []; + } + + return store.notes.filter(n => + n.post.id !== note.post.id && n.post.created_at <= note.post.created_at, + ); + }; + const replyNotes: () => PrimalNote[] = () => { + const note = primaryNote(); + + if (!note) { + return []; + } + + return store.notes.filter(n => + n.post.id !== note.post.id && n.post.created_at >= note.post.created_at, + ); + }; + + const [store, updateStore] = createStore({ + ...initialData, + actions: { + saveNotes, + fetchNotes, + clearNotes, + fetchNextPage, + updatePage, + savePage, + setPrimaryNote, + }, + }); + +// RENDER --------------------------------------- + + return ( + + {props.children} + + ); +} + +export const useThreadContext = () => useContext(ThreadContext); diff --git a/src/contexts/TranslatorContext.tsx b/src/contexts/TranslatorContext.tsx new file mode 100644 index 0000000..ddafdab --- /dev/null +++ b/src/contexts/TranslatorContext.tsx @@ -0,0 +1,49 @@ +import { createStore } from "solid-js/store"; +import { + createContext, + JSXElement, + useContext +} from "solid-js"; +import { IntlProvider } from "@cookbook/solid-intl"; + + +export type TranslatorContextStore = { + locale: string, + messages: Record, + actions: { + setLocale: (locale: string) => void, + }, +} + +const initialData = { + locale: 'en', + messages: {}, +}; + +export const TranslatorContext = createContext(); + +export function TranslatorProvider(props: { children: JSXElement }) { + + const setLocale = (locale: string) => { + updateStore('locale', () => locale); + }; + +// STORES --------------------------------------- + +const [store, updateStore] = createStore({ + ...initialData, + actions: { + setLocale, + }, +}); + + return ( + + + {props.children} + + + ); +} + +export function useTranslatorContext() { return useContext(TranslatorContext); } diff --git a/src/formats.ts b/src/formats.ts new file mode 100644 index 0000000..57e4878 --- /dev/null +++ b/src/formats.ts @@ -0,0 +1,9 @@ +type NumberFormatOptions = { + style?: "unit" | "currency" | "decimal" | "percent", + unit?: string, + unitDisplay?: "narrow" | "long" | "short", +}; + +export const hourNarrow: NumberFormatOptions = { + style: 'unit', unit: 'hour', unitDisplay: 'narrow', +}; diff --git a/src/index.scss b/src/index.scss new file mode 100644 index 0000000..bb6d972 --- /dev/null +++ b/src/index.scss @@ -0,0 +1,462 @@ +@import "@picocss/pico/scss/pico"; + +@mixin sunset_wave { + --brand-1: #FA3C3C; + --brand-2: #5B09AD; + --brand-3: #FA9A43; + --brand-gradient: linear-gradient(135deg, var(--brand-1) 0%, var(--brand-2) 100%); + --brand-gradient-vertical: linear-gradient(180deg, var(--brand-1) 0%, var(--brand-2) 100%); + --highlight-gradient: linear-gradient(137.63deg, var(--brand-3) 0%, var(--brand-1) 32.3%, var(--brand-2) 100%); + --accent-1: #CA079F; + --accent-2: #AB268E; + + --brand-text: #D5D5D5; + --background-site: #000000; + --background-card: #121212; + --background-input: #222222; + --background-modal: #00000066; + --background-embedded_card: #1A1A1A; + --border-embedded-card: #282828; + --text-primary: #FFFFFF; + --text-secondary: #AAAAAA; + --text-secondary-2: #AAAAAA; + --text-tertiary: #757575; + --text-tertiary-2: #666666; + --subtile-devider: #444444; + --check-image: url('./assets/icons/check.svg'); + + --primary: var(--accent-1); + --primary-hover: var(--accent-2); + --primary-focus: var(--background-color); + --background-color: var(--background-site); + + --active-link: var(--text-primary); + --inactive-link: var(--text-secondary); + --fade-gradient-vertical: linear-gradient(180deg, #000000FF 80%, #00000000 100%); + --fade-note-vertical: linear-gradient(180deg, #12121200, #121212FF 90%); + + --logo: url('./assets/icons/logo_fire.svg'); + --icon-follows: url('./assets/icons/follows.svg'); + --icon-tribe: url('./assets/icons/tribe.svg'); + --icon-global: url('./assets/icons/global.svg'); + --icon-network: url('./assets/icons/network.svg'); + + --icon-follows-latest: url('./assets/icons/follows_latest.svg'); + --icon-follows-popular: url('./assets/icons/follows_popular.svg'); + --icon-follows-trending: url('./assets/icons/follows_trending.svg'); + + --icon-tribe-latest: url('./assets/icons/tribe_latest.svg'); + --icon-tribe-popular: url('./assets/icons/tribe_popular.svg'); + --icon-tribe-trending: url('./assets/icons/tribe_trending.svg'); + + --icon-global-latest: url('./assets/icons/global_latest.svg'); + --icon-global-popular: url('./assets/icons/global_popular.svg'); + --icon-global-trending: url('./assets/icons/global_trending.svg'); + + --icon-network-latest: url('./assets/icons/network_latest.svg'); + --icon-network-popular: url('./assets/icons/network_popular.svg'); + --icon-network-trending: url('./assets/icons/network_trending.svg'); + + + select { + background-color: var(--background-site); + } + +} + +@mixin sunrise_wave { + --brand-1: #FA3C3C; + --brand-2: #5B09AD; + --brand-3: #5B09AD; + --brand-gradient: linear-gradient(135deg, var(--brand-1) 0%, var(--brand-2) 100%); + --brand-gradient-vertical: linear-gradient(180deg, var(--brand-1) 0%, var(--brand-2) 100%); + --highlight-gradient: linear-gradient(137.63deg, var(--brand-3) 0%, var(--brand-1) 32.3%, var(--brand-2) 100%); + --accent-1: #CA079F; + --accent-2: #AB268E; + + --brand-text: #444444; + --background-site: #F5F5F5; + --background-card: #FFFFFF; + --background-input: #E5E5E5; + --background-modal: #F5F5F566; + --background-embedded_card: #F5F5F5; + --border-embedded-card: #E5E5E5; + --text-primary: #111111; + --text-secondary: #444444; + --text-secondary-2: #666666; + --text-tertiary: #808080; + --text-tertiary-2: #808080; + --subtile-devider: #C8C8C8; + --check-image: url('./assets/icons/check-black.svg'); + + --primary: var(--accent-1); + --primary-hover: var(--accent-2); + --primary-focus: var(--background-color); + --background-color: var(--background-site); + + --active-link: var(--text-primary); + --inactive-link: var(--text-secondary); + --fade-gradient-vertical: linear-gradient(180deg, #F5F5F5FF 80%, #F5F5F500 100%); + --fade-note-vertical: linear-gradient(180deg, #FFFFFF00, #FFFFFFFF 90%); + + --logo: url('./assets/icons/logo_fire.svg'); + --icon-follows: url('./assets/icons/follows_light.svg'); + --icon-tribe: url('./assets/icons/tribe_light.svg'); + --icon-global: url('./assets/icons/global_light.svg'); + --icon-network: url('./assets/icons/network_light.svg'); + + --icon-follows-latest: url('./assets/icons/follows_latest_light.svg'); + --icon-follows-popular: url('./assets/icons/follows_popular_light.svg'); + --icon-follows-trending: url('./assets/icons/follows_trending_light.svg'); + + --icon-tribe-latest: url('./assets/icons/tribe_latest_light.svg'); + --icon-tribe-popular: url('./assets/icons/tribe_popular_light.svg'); + --icon-tribe-trending: url('./assets/icons/tribe_trending_light.svg'); + + --icon-global-latest: url('./assets/icons/global_latest_light.svg'); + --icon-global-popular: url('./assets/icons/global_popular_light.svg'); + --icon-global-trending: url('./assets/icons/global_trending_light.svg'); + + --icon-network-latest: url('./assets/icons/network_latest_light.svg'); + --icon-network-popular: url('./assets/icons/network_popular_light.svg'); + --icon-network-trending: url('./assets/icons/network_trending_light.svg'); + + select { + background-color: var(--background-site); + } +} + +@mixin midnight_wave { + --brand-1: #0090F8; + --brand-2: #4C00C7; + --brand-3: #00E0FF; + --brand-gradient: linear-gradient(135deg, var(--brand-1) 0%, var(--brand-2) 100%); + --brand-gradient-vertical: linear-gradient(180deg, var(--brand-1) 0%, var(--brand-2) 100%); + --highlight-gradient: linear-gradient(137.63deg, var(--brand-3) 0%, var(--brand-1) 32.22%, var(--brand-2) 100%); + + --accent-1: #2394EF; + --accent-2: #0C7DD8; + + --brand-text: #D5D5D5; + --background-site: #000000; + --background-card: #121212; + --background-input: #222222; + --background-modal: #00000066; + --background-embedded_card: #1A1A1A; + --border-embedded-card: #282828; + --text-primary: #FFFFFF; + --text-secondary: #AAAAAA; + --text-secondary-2: #AAAAAA; + --text-tertiary: #757575; + --text-tertiary-2: #666666; + --subtile-devider: #444444; + --card-border: #282828; + --check-image: url('./assets/icons/check.svg'); + + --primary: var(--accent-1); + --primary-hover: var(--accent-2); + --primary-focus: var(--background-color); + --background-color: var(--background-site); + + --active-link: var(--text-primary); + --inactive-link: var(--text-secondary); + --fade-gradient-vertical: linear-gradient(180deg, #000000FF 80%, #00000000 100%); + --fade-note-vertical: linear-gradient(180deg, #12121200, #121212FF 90%); + + --logo: url('./assets/icons/logo_ice.svg'); + --icon-follows: url('./assets/icons/follows.svg'); + --icon-tribe: url('./assets/icons/tribe.svg'); + --icon-global: url('./assets/icons/global.svg'); + --icon-network: url('./assets/icons/network.svg'); + + --icon-follows-latest: url('./assets/icons/follows_latest.svg'); + --icon-follows-popular: url('./assets/icons/follows_popular.svg'); + --icon-follows-trending: url('./assets/icons/follows_trending.svg'); + + --icon-tribe-latest: url('./assets/icons/tribe_latest.svg'); + --icon-tribe-popular: url('./assets/icons/tribe_popular.svg'); + --icon-tribe-trending: url('./assets/icons/tribe_trending.svg'); + + --icon-global-latest: url('./assets/icons/global_latest.svg'); + --icon-global-popular: url('./assets/icons/global_popular.svg'); + --icon-global-trending: url('./assets/icons/global_trending.svg'); + + --icon-network-latest: url('./assets/icons/network_latest.svg'); + --icon-network-popular: url('./assets/icons/network_popular.svg'); + --icon-network-trending: url('./assets/icons/network_trending.svg'); + + + select { + background-color: var(--background-site); + } +} + + +@mixin ice_wave { + --brand-1: #0090F8; + --brand-2: #4C00C7; + --brand-3: #00E0FF; + --brand-gradient: linear-gradient(135deg, var(--brand-1) 0%, var(--brand-2) 100%); + --brand-gradient-vertical: linear-gradient(180deg, var(--brand-1) 0%, var(--brand-2) 100%); + --highlight-gradient: linear-gradient(137.63deg, var(--brand-3) 0%, var(--brand-1) 32.22%, var(--brand-2) 100%); + --accent-1: #2394EF; + --accent-2: #0C7DD8; + + --brand-text: #444444; + --background-site: #F5F5F5; + --background-card: #FFFFFF; + --background-input: #E5E5E5; + --background-modal: #F5F5F566; + --background-embedded_card: #F5F5F5; + --border-embedded-card: #E5E5E5; + --text-primary: #111111; + --text-secondary: #444444; + --text-secondary-2: #666666; + --text-tertiary: #808080; + --text-tertiary-2: #808080; + --subtile-devider: #C8C8C8; + --check-image: url('./assets/icons/check-black.svg'); + + --primary: var(--accent-1); + --primary-hover: var(--accent-2); + --primary-focus: var(--background-color); + --background-color: var(--background-site); + + --active-link: var(--text-primary); + --inactive-link: var(--text-secondary); + --fade-gradient-vertical: linear-gradient(180deg, #F5F5F5FF 80%, #F5F5F500 100%); + --fade-note-vertical: linear-gradient(180deg, #FFFFFF00, #FFFFFFFF 90%); + + --logo: url('./assets/icons/logo_ice.svg'); + --icon-follows: url('./assets/icons/follows_light.svg'); + --icon-tribe: url('./assets/icons/tribe_light.svg'); + --icon-global: url('./assets/icons/global_light.svg'); + --icon-network: url('./assets/icons/network_light.svg'); + + --icon-follows-latest: url('./assets/icons/follows_latest_light.svg'); + --icon-follows-popular: url('./assets/icons/follows_popular_light.svg'); + --icon-follows-trending: url('./assets/icons/follows_trending_light.svg'); + + --icon-tribe-latest: url('./assets/icons/tribe_latest_light.svg'); + --icon-tribe-popular: url('./assets/icons/tribe_popular_light.svg'); + --icon-tribe-trending: url('./assets/icons/tribe_trending_light.svg'); + + --icon-global-latest: url('./assets/icons/global_latest_light.svg'); + --icon-global-popular: url('./assets/icons/global_popular_light.svg'); + --icon-global-trending: url('./assets/icons/global_trending_light.svg'); + + --icon-network-latest: url('./assets/icons/network_latest_light.svg'); + --icon-network-popular: url('./assets/icons/network_popular_light.svg'); + --icon-network-trending: url('./assets/icons/network_trending_light.svg'); + select { + background-color: var(--background-site); + } +} + +/* Default theme */ +:root[data-theme="dark"], +:root[data-theme="sunset"], +:root:not([data-theme="dark"]), +:root:not([data-theme="sunset"]), +:root:not([data-theme="sunrise"]), +:root:not([data-theme="midinght"]), +:root:not([data-theme="ice"]) { + @include sunset_wave(); +} + +:root[data-theme="light"], +:root[data-theme="sunrise"] { + @include sunrise_wave(); +} + +:root[data-theme="midnight"] { + @include midnight_wave(); +} + +:root[data-theme="ice"] { + @include ice_wave(); +} + +/* Automatically enabled if user has Dark mode enabled */ +@media only screen and (prefers-color-scheme: dark) { + :root { + @include sunset_wave(); + } +} + +/* Common styles */ +:root { + --missing-avatar-text: #FFFFFF; + + --z-index-lifted: 10; + --z-index-header: 20; + --z-index-floater: 30; + + --sidebar-section-icon-gradient: linear-gradient(175.11deg, #FA9A43 6.94%, #FA4343 29.79%, #5B12A4 97.76%), linear-gradient(170.29deg, #CCCCCC 12.73%, #808080 94.98%), #D9D9D9; + + --light-input: #E5E5E5; + --light-back: #F5F5F5; + --dark-input: #222222; + --dark-back: #000000; + + .mentioned_user { + color: var(--accent-1); + } + .hash_tag { + color: var(--accent-1); + } + + .postImage { + display: block; + width: 100%; + border-radius: 12px; + } + + .w-max { + width: 100%; + height: 300px; + border-radius: 4px; + } + + * { + ::-moz-selection { + color: var(--background-site); + background: var(--text-primary); + } + + ::selection { + color: var(--background-site); + background: var(--text-primary); + } + } + +} + +body { + margin: 0; + font-family: 'Roboto Flex', sans-serif; + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; + overflow-y: scroll; + background-color: var(--background-site); +} + +.linkish { + color: var(--accent-1); +} + +.reply_icon { + -webkit-mask: url(./assets/icons/feed_reply.svg) no-repeat 0 / 100%; + mask: url(./assets/icons/feed_reply_fill.svg) no-repeat 0 / 100%; +} +.reply_icon { + -webkit-mask: url(./assets/icons/feed_reply_fill.svg) no-repeat 0 / 100%; + mask: url(./assets/icons/feed_reply_fill.svg) no-repeat 0 / 100%; +} +.repost_icon { + -webkit-mask: url(./assets/icons/feed_repost.svg) no-repeat 0 / 100%; + mask: url(./assets/icons/feed_repost_fill.svg) no-repeat 0 / 100%; +} +.repost_icon { + -webkit-mask: url(./assets/icons/feed_repost_fill.svg) no-repeat 0 / 100%; + mask: url(./assets/icons/feed_repost_fill.svg) no-repeat 0 / 100%; +} +.zap_icon { + -webkit-mask: url(./assets/icons/feed_zap.svg) no-repeat 0 / 100%; + mask: url(./assets/icons/feed_zap_fill.svg) no-repeat 0 / 100%; +} +.zap_icon { + -webkit-mask: url(./assets/icons/feed_zap_fill.svg) no-repeat 0 / 100%; + mask: url(./assets/icons/feed_zap_fill.svg) no-repeat 0 / 100%; +} +.like_icon { + -webkit-mask: url(./assets/icons/feed_like.svg) no-repeat 0 / 100%; + mask: url(./assets/icons/feed_like_fill.svg) no-repeat 0 / 100%; +} +.like_icon { + -webkit-mask: url(./assets/icons/feed_like_fill.svg) no-repeat 0 / 100%; + mask: url(./assets/icons/feed_like_fill.svg) no-repeat 0 / 100%; +} +.attach_icon { + -webkit-mask: url(./assets/icons/attach_media.svg) no-repeat 0 / 100%; + mask: url(./assets/icons/attach_media.svg) no-repeat 0 / 100%; +} + +// Scrollbars + +/* width */ +::-webkit-scrollbar { + width: 6px; + height: 6px; +} + +/* Track */ +::-webkit-scrollbar-track { + background: none; +} + +/* Handle */ +::-webkit-scrollbar-thumb { + background: var(--subtile-devider); + border-radius: 3px; +} + +/* Handle on hover */ +::-webkit-scrollbar-thumb:hover { + background: var(--text-tertiary-2); +} + +// Checkboxes + +[type="checkbox"] { + background-color: var(--background-card); + border-color: var(--text-tertiary-2); + -webkit-appearance: none; + -moz-appearance: none; + appearance: none; + width: 20px; + height: 20px; + margin: 0; + margin-inline-end: 16px; + border-width: 1px; + font-size: inherit; + vertical-align: middle; + cursor: pointer; + + &::-ms-check { + display: none; // unstyle IE checkboxes + } + + &:checked, + &:checked:active, + &:checked:focus { + background-color: var(--background-card); + border-color: var(--text-tertiary-2); + background-position: center; + background-size: 12px auto; + background-repeat: no-repeat; + background-image: var(--check-image); + } + + &:focus { + background-color: var(--background-card); + border-color: var(--text-tertiary-2); + } + + & ~ label { + display: inline-block; + margin-right: 0; + margin-bottom: 0; + cursor: pointer; + } + + &:indeterminate { + background-color: var(--background-site); + border-color: var(--text-tertiary-2); + background-image: var(--icon-minus); + background-position: center; + background-size: 12px auto; + background-repeat: no-repeat; + } + +} diff --git a/src/index.tsx b/src/index.tsx new file mode 100644 index 0000000..a1a9779 --- /dev/null +++ b/src/index.tsx @@ -0,0 +1,8 @@ +/* @refresh reload */ +import { render } from 'solid-js/web'; + +import './index.scss'; +import App from './App'; +import { Router } from '@solidjs/router'; + +render(() => , document.getElementById('root') as HTMLElement); diff --git a/src/lib/availableFeeds.ts b/src/lib/availableFeeds.ts new file mode 100644 index 0000000..bffcc09 --- /dev/null +++ b/src/lib/availableFeeds.ts @@ -0,0 +1,69 @@ +import { defaultFeeds, noKey } from "../constants"; +import { PrimalFeed } from "../types/primal"; +import { getStorage, saveFeeds } from "./localStore"; + + +export const initAvailableFeeds = (pubkey: string | undefined) => { + const storage = getStorage(pubkey); + + if (storage.feeds && storage.feeds.length === 0) { + saveFeeds(pubkey, defaultFeeds); + return defaultFeeds; + } + + return storage.feeds; +} + +export const updateAvailableFeedsTop = ( + pubkey: string | undefined, + feed: PrimalFeed, + feeds: PrimalFeed[], +) => { + if (feeds.find(f => feed.name === f.name)) { + return [...feeds]; + } + + const newFeeds = [ { ...feed }, ...feeds]; + + saveFeeds(pubkey, newFeeds); + + return newFeeds; +}; + +export const updateAvailableFeeds = ( + pubkey: string | undefined, + feed: PrimalFeed, + feeds: PrimalFeed[], +) => { + if (feeds.find(f => feed.name === f.name)) { + return [...feeds]; + } + + const newFeeds = [ ...feeds, { ...feed }]; + + saveFeeds(pubkey, newFeeds); + + return newFeeds; +}; + +export const removeFromAvailableFeeds = ( + pubkey: string | undefined, + feed: PrimalFeed, + feeds: PrimalFeed[], +) => { + const newFeeds = feeds.filter(f => f.hex !== feed.hex); + + saveFeeds(pubkey, newFeeds); + + return newFeeds; +}; + +export const replaceAvailableFeeds = ( + pubkey: string | undefined, + feeds: PrimalFeed[], +) => { + const newFeeds = [...feeds]; + saveFeeds(pubkey, newFeeds); + + return newFeeds; +} diff --git a/src/lib/dates.ts b/src/lib/dates.ts new file mode 100644 index 0000000..c5128dc --- /dev/null +++ b/src/lib/dates.ts @@ -0,0 +1,57 @@ +export const shortDate = (timestamp: number | undefined) => { + if (!timestamp || timestamp < 0) { + return ''; + } + const date = new Date(timestamp * 1000); + const dtf = new Intl.DateTimeFormat('en-US', { dateStyle: 'medium'}); + + return dtf.format(date); +}; + +export const date = (postTimestamp: number, style: Intl.RelativeTimeFormatStyle = 'short', since?: number) => { + const today = since ?? Math.floor((new Date()).getTime() / 1000); + const date = new Date(postTimestamp * 1000); + + const minute = 60; + const hour = minute * 60; + const day = hour * 24; + const week = day * 7; + const month = day * 30; + const year = month * 12; + + const rtf = new Intl.RelativeTimeFormat('en', { style }); + + const diff = today - postTimestamp; + + if ( diff > year) { + const years = Math.floor(diff / year); + return { date, label: rtf.format(-years, 'years') }; + } + + if (diff > month) { + const months = Math.floor(diff / month); + return { date, label: rtf.format(-months, 'months') }; + } + + if (diff > week) { + const weeks = Math.floor(diff / week); + return { date, label: rtf.format(-weeks, 'weeks') }; + } + + if (diff > day) { + const days = Math.floor(diff / day); + return { date, label: rtf.format(-days, 'days') }; + } + + if (diff > hour) { + const hours = Math.floor(diff / hour); + return { date, label: rtf.format(-hours, 'hours') }; + } + + if (diff > minute) { + const minutes = Math.floor(diff / minute); + return { date, label: rtf.format(-minutes, 'minutes') }; + } + + return { date, label: `${diff}s` }; +}; diff --git a/src/lib/feed.ts b/src/lib/feed.ts new file mode 100644 index 0000000..25166e9 --- /dev/null +++ b/src/lib/feed.ts @@ -0,0 +1,224 @@ +import { sendMessage } from "../sockets"; +import { ExploreFeedPayload } from "../types/primal"; +import { nip19 } from "nostr-tools"; +import { day, hour, noKey } from "../constants"; + +export const getFutureFeed = (user_pubkey: string | undefined, pubkey: string | undefined, subid: string, since: number) => { + if (!pubkey || pubkey === noKey) { + return; + } + + let payload: { since: number, pubkey: string, user_pubkey?: string, limit: number } = + { since, pubkey, limit: 1000 }; + + if (user_pubkey && user_pubkey !== noKey) { + payload.user_pubkey = user_pubkey; + } + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["feed", payload]}, + ])); +}; + +export const getFeed = (user_pubkey: string | undefined, pubkey: string | undefined, subid: string, until = 0, limit = 20) => { + if (!pubkey || pubkey === noKey) { + return; + } + const start = until === 0 ? 'since' : 'until'; + + let payload = { limit, [start]: until, pubkey }; + + if (user_pubkey && user_pubkey !== noKey) { + payload.user_pubkey = user_pubkey; + } + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["feed", payload]}, + ])); +} + +export const getEvents = (user_pubkey: string | undefined, eventIds: string[], subid: string, extendResponse?: boolean) => { + + let payload: {event_ids: string[], user_pubkey?: string, extended_response?: boolean } = + { event_ids: eventIds } ; + + if (user_pubkey && user_pubkey !== noKey) { + payload.user_pubkey = user_pubkey; + } + + if (extendResponse) { + payload.extended_response = extendResponse; + } + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["events", payload]}, + ])); + +}; + +export const getUserFeed = (user_pubkey: string | undefined, pubkey: string | undefined, subid: string, until = 0, limit = 20) => { + if (!pubkey || pubkey === noKey) { + return; + } + + const start = until === 0 ? 'since' : 'until'; + + let payload = { pubkey, limit, notes: 'authored', [start]: until } ; + + if (user_pubkey && user_pubkey !== noKey) { + payload.user_pubkey = user_pubkey; + } + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["feed", payload]}, + ])); +} + +export const getThread = (user_pubkey: string | undefined, postId: string, subid: string, until = 0, limit = 20) => { + + const decoded = nip19.decode(postId).data; + let event_id = ''; + + + if (typeof decoded === 'string') { + event_id = decoded; + } + + if (typeof decoded !== 'string' && 'id' in decoded) { + event_id = decoded.id; + } + + if (event_id.length === 0) { + return; + } + + let payload: { user_pubkey?: string, limit: number, event_id: string, until?: number } = + { event_id, limit: 100 } ; + + if (user_pubkey && user_pubkey !== noKey) { + payload.user_pubkey = user_pubkey; + } + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["thread_view", payload]}, + ])); +} + +export const getFutureExploreFeed = ( + user_pubkey: string | undefined, + subid: string, + scope: string, + timeframe: string, + since: number, + ) => { + + let payload: { timeframe: string, scope: string, since: number, user_pubkey?: string, created_after?: number, limit: number } = + { timeframe, scope, since, limit: 1000, }; + + if (user_pubkey && user_pubkey !== noKey) { + payload.user_pubkey = user_pubkey; + } + + if (since > 0) { + payload.since = since; + } + + if (timeframe === 'trending') { + const yesterday = Math.floor((new Date().getTime() - day) / 1000); + + payload.created_after = yesterday; + } + + if (timeframe === 'mostzapped4h') { + const fourHAgo = Math.floor((new Date().getTime() - (4 * hour)) / 1000); + + payload.timeframe = 'mostzapped'; + payload.created_after = fourHAgo; + } + + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: [ + "explore", + payload, + ]}, + ])); +}; + +export const getExploreFeed = ( + pubkey: string | undefined, + subid: string, + scope: string, + timeframe: string, + until = 0, + limit = 20, +) => { + + let payload: ExploreFeedPayload = { timeframe, scope, limit }; + + if (pubkey && pubkey !== noKey) { + payload.user_pubkey = pubkey; + } + + if (until > 0) { + payload.until = until; + } + + if (timeframe === 'trending') { + const yesterday = Math.floor((new Date().getTime() - day) / 1000); + + payload.created_after = yesterday; + } + + if (timeframe === 'mostzapped4h') { + const fourHAgo = Math.floor((new Date().getTime() - (4 * hour)) / 1000); + + payload.timeframe = 'mostzapped'; + payload.created_after = fourHAgo; + } + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: [ + "explore", + payload, + ]}, + ])); +}; + +export const getTrending24h = ( + subid: string, +) => { + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: [ + "explore_global_trending_24h", + ]}, + ])); +}; + +export const getMostZapped4h = ( + subid: string, +) => { + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: [ + "explore_global_mostzapped_4h", + ]}, + ])); +}; diff --git a/src/lib/keys.ts b/src/lib/keys.ts new file mode 100644 index 0000000..b0812bb --- /dev/null +++ b/src/lib/keys.ts @@ -0,0 +1,6 @@ +import { nip19 } from "nostr-tools" +import { noKey } from "../constants"; + +export const hexToNpub = (hex: string | undefined): string => { + return hex && hex !== noKey ? nip19.npubEncode(hex) : ''; +} diff --git a/src/lib/localStore.ts b/src/lib/localStore.ts new file mode 100644 index 0000000..d484703 --- /dev/null +++ b/src/lib/localStore.ts @@ -0,0 +1,111 @@ +import { noKey } from "../constants" +import { NostrRelays, PrimalFeed } from "../types/primal"; + +export type LocalStore = { + following: string[], + followingSince: number, + relaySettings: NostrRelays, + likes: string[], + feeds: PrimalFeed[]; + theme: string, +}; + +export const emptyStorage = { + following: [], + followingSince: 0, + relaySettings: {}, + likes: [], + feeds: [], + theme: 'sunset', +} + +export const storageName = (pubkey?: string) => { + if (!pubkey || pubkey === noKey) { + return 'anon'; + } + + return `store_${pubkey}`; +}; + +export const getStorage = (pubkey?: string) => { + if (!pubkey) { + return {} as LocalStore; + } + + const name = storageName(pubkey); + const storage = localStorage.getItem(name); + + return storage ? + JSON.parse(storage) as LocalStore : + { ...emptyStorage }; +}; + +export const setStorage = (pubkey: string | undefined, data: LocalStore) => { + if (!pubkey) { + return; + } + + const name = storageName(pubkey); + const value = JSON.stringify(data); + + localStorage.setItem(name, value); +} + +export const saveFollowing = (pubkey: string | undefined, following: string[], since: number) => { + if (!pubkey) { + return; + } + + const store = getStorage(pubkey); + + store.following = [...following]; + store.followingSince = since; + + setStorage(pubkey, store); +} + +export const saveRelaySettings = (pubkey: string | undefined, settings: NostrRelays) => { + if (!pubkey) { + return; + } + + const store = getStorage(pubkey); + + store.relaySettings = { ...settings }; + + setStorage(pubkey, store); +} + +export const saveLikes = (pubkey: string | undefined, likes: string[]) => { + if (!pubkey) { + return; + } + + const store = getStorage(pubkey); + + store.likes = [ ...likes ]; + + setStorage(pubkey, store); +}; + +export const saveFeeds = (pubkey: string | undefined, feeds: PrimalFeed[]) => { + if (!pubkey) { + return; + } + const store = getStorage(pubkey); + + store.feeds = [ ...feeds ]; + + setStorage(pubkey, store); +}; + +export const saveTheme = (pubkey: string | undefined, theme: string) => { + if (!pubkey) { + return; + } + const store = getStorage(pubkey); + + store.theme = theme; + + setStorage(pubkey, store); +}; diff --git a/src/lib/media.ts b/src/lib/media.ts new file mode 100644 index 0000000..6d70af1 --- /dev/null +++ b/src/lib/media.ts @@ -0,0 +1,50 @@ +import { Kind } from "../constants"; +import { sendMessage } from "../uploadSocket"; +import { NostrWindow } from "../types/primal"; + +export const getMediaUrl = (url: string | undefined, size = 'o', animated = 1) => { + if (!url) { + return; + } + const mediaServer = localStorage.getItem('mediaServer'); + + if (!mediaServer) { + return url; + } + + const encodedUrl = encodeURIComponent(url); + + return `${mediaServer}/media-cache?s=${size}&a=${animated}&u=${encodedUrl}`; +} + +export const uploadMedia = async ( + uploader: string | undefined, + subid: string, + content: string, +) => { + if (!uploader) { + return; + } + + const event = { + content, + kind: Kind.Upload, + tags: [['p', uploader]], + created_at: Math.floor((new Date()).getTime() / 1000), + }; + + const win = window as NostrWindow; + const nostr = win.nostr; + + if (nostr === undefined) { + return false; + } + + const signedNote = await nostr.signEvent(event); + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["upload", { event_from_user: signedNote }]}, + ])); +}; diff --git a/src/lib/messages.ts b/src/lib/messages.ts new file mode 100644 index 0000000..8790efc --- /dev/null +++ b/src/lib/messages.ts @@ -0,0 +1,96 @@ +import { Kind } from "../constants"; +import { sendMessage } from "../sockets"; +import { NostrWindow, UserRelation } from "../types/primal"; + + +export const subscribeToMessagesStats = (pubkey: string, subid: string) => { + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["directmsg_count", { pubkey }]}, + ])); +} + +export const resetMessageCount = async (sender: string, subid: string) => { + + const win = window as NostrWindow; + const nostr = win.nostr; + + if (nostr === undefined) { + return false; + } + + const event = { + content: `{ "description": "reset messages from '${sender}'"}`, + kind: Kind.Settings, + tags: [["d", "Primal-Web App"]], + created_at: Math.ceil((new Date()).getTime() / 1000), + }; + + const signedEvent = await nostr.signEvent(event); + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["reset_directmsg_count", { + event_from_user: signedEvent, + sender, + }]}, + ])); +} + +export const getMessageCounts = (user_pubkey: string, relation: UserRelation, subid: string) => { + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["get_directmsg_contacts", { user_pubkey, relation }]}, + ])); +} + +export const getOldMessages = (receiver: string, sender: string, subid: string, until = 0, limit = 20) => { + + const start = until === 0 ? 'since' : 'until'; + + const payload = { limit, [start]: until, receiver, sender }; + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["get_directmsgs", payload]}, + ])); +} + +export const getNewMessages = (receiver: string, sender: string, subid: string, since = 0, limit = 20) => { + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["get_directmsgs", { receiver, sender, since, limit }]}, + ])); +} + +export const markAllAsRead = async (subid: string) => { + + const win = window as NostrWindow; + const nostr = win.nostr; + + if (nostr === undefined) { + return false; + } + + const event = { + content: `{ "description": "mark all messages as read"}`, + kind: Kind.Settings, + tags: [["d", "Primal-Web App"]], + created_at: Math.ceil((new Date()).getTime() / 1000), + }; + + const signedEvent = await nostr.signEvent(event); + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["reset_directmsg_counts", { + event_from_user: signedEvent, + }]}, + ])); +} diff --git a/src/lib/notes.ts b/src/lib/notes.ts new file mode 100644 index 0000000..ba6c5b8 --- /dev/null +++ b/src/lib/notes.ts @@ -0,0 +1,317 @@ +import { getLinkPreview } from "link-preview-js"; +import { Relay } from "nostr-tools"; +import { createStore } from "solid-js/store"; +import { Kind } from "../constants"; +import { NostrWindow, PrimalNote } from "../types/primal"; +import { getMediaUrl } from "./media"; + +const getLikesStorageKey = () => { + const key = localStorage.getItem('pubkey') || 'anon'; + return `likes_${key}`; +}; + +export const getStoredLikes = () => { + return JSON.parse(localStorage.getItem(getLikesStorageKey()) || '[]'); +}; + +export const setStoredLikes = (likes: string[]) => { + return localStorage.setItem(getLikesStorageKey(), JSON.stringify(likes)); +}; + +export const sanitize = (html: string) => { + return html.replaceAll('<', '<').replaceAll('>', '>'); +}; + +export const [linkPreviews, setLinkPreviews] = createStore>({}); + +export const addLinkPreviews = async (url: string) => { + try { + const preview = await getLinkPreview(url); + + // const preview = await fetch(`link-preview?u=https://yahoo.com`); + // console.log('PREV: ', preview); + setLinkPreviews((p) => ({ ...p, [url]: { ...preview }})); + + } catch (e) { + console.log('Failed to get preview for: ', url); + setLinkPreviews((p) => ({ ...p, [url]: { noPreview: url }})); + } +}; + +export const spotifyRegex = /open\.spotify\.com\/(track|album|playlist|episode)\/([a-zA-Z0-9]+)/; +export const twitchRegex = /twitch.tv\/([a-z0-9_]+$)/i; +export const mixCloudRegex = /mixcloud\.com\/(?!live)([a-zA-Z0-9]+)\/([a-zA-Z0-9-]+)/; +// export const tidalRegex = /tidal\.com\/(?:browse\/)?(\w+)\/([a-z0-9-]+)/i; +export const soundCloudRegex = /soundcloud\.com\/(?!live)([a-zA-Z0-9]+)\/([a-zA-Z0-9-]+)/; +// export const tweetUrlRegex = /https?:\/\/twitter\.com\/(?:#!\/)?(\w+)\/status(?:es)?\/(\d+)/; +export const appleMusicRegex = /music\.apple\.com\/([a-z]{2}\/)?(?:album|playlist)\/[\w\d-]+\/([.a-zA-Z0-9-]+)(?:\?i=\d+)?/i; +export const nostrNestsRegex = /nostrnests\.com\/[a-zA-Z0-9]+/i; +// export const magnetRegex = /(magnet:[\S]+)/i; +export const wavlakeRegex = /(?:player\.)?wavlake\.com\/(track\/[.a-zA-Z0-9-]+|album\/[.a-zA-Z0-9-]+|[.a-zA-Z0-9-]+)/i; +// export const odyseeRegex = /odysee\.com\/([a-zA-Z0-9]+)/; +export const youtubeRegex = /(?:https?:\/\/)?(?:www|m\.)?(?:youtu\.be\/|youtube\.com\/(?:live\/|shorts\/|embed\/|v\/|watch\?v=|watch\?.+&v=))((\w|-){11})/; + +export const urlify = (text: string, highlightOnly = false, skipEmbed = false, skipLinkPreview = false) => { + const urlRegex = /https?:\/\/(www\.)?[-a-zA-Z0-9@:%._\+~#=]{1,256}\.[a-zA-Z0-9()]{1,8}\b([-a-zA-Z0-9()@:%_\+.~#?&//=]*)/g; + + return text.replace(urlRegex, (url) => { + if (!skipEmbed) { + + const isImage = url.includes('.jpg')|| url.includes('.jpeg')|| url.includes('.webp') || url.includes('.png') || url.includes('.gif') || url.includes('format=png'); + + if (isImage) { + return '' + } + + const isMp4Video = url.includes('.mp4') || url.includes('.mov'); + + if (isMp4Video) { + return ``; + } + + const isOggVideo = url.includes('.ogg'); + + if (isOggVideo) { + return ``; + } + + const isWebmVideo = url.includes('.webm'); + + if (isWebmVideo) { + return ``; + } + + if (youtubeRegex.test(url)) { + const youtubeId = youtubeRegex.test(url) && RegExp.$1; + + return ``; + } + + if (spotifyRegex.test(url)) { + const convertedUrl = url.replace(/\/(track|album|playlist|episode)\/([a-zA-Z0-9]+)/, "/embed/$1/$2"); + + return ``; + } + + if (twitchRegex.test(url)) { + const channel = url.split("/").slice(-1); + + const args = `?channel=${channel}&parent=${window.location.hostname}&muted=true`; + return ``; + } + + if (mixCloudRegex.test(url)) { + const feedPath = (mixCloudRegex.test(url) && RegExp.$1) + "%2F" + (mixCloudRegex.test(url) && RegExp.$2); + + // const lightTheme = useLogin().preferences.theme === "light"; + // const lightParams = lightTheme ? "light=1" : "light=0"; + return ` +
+ `; + } + + if (soundCloudRegex.test(url)) { + return ``; + } + + if (appleMusicRegex.test(url)) { + const convertedUrl = url.replace("music.apple.com", "embed.music.apple.com"); + const isSongLink = /\?i=\d+$/.test(convertedUrl); + + return ` + + `; + } + + if (nostrNestsRegex.test(url)) { + return ` + `; + } + + if (wavlakeRegex.test(url)) { + const convertedUrl = url.replace(/(?:player\.|www\.)?wavlake\.com/, "embed.wavlake.com"); + + return ` + `; + } + } + + if (highlightOnly) { + return `${url}`; + } + + if (skipLinkPreview) { + return `${url}`; + } + + addLinkPreviews(url); + + return `__LINK__${url}__LINK__`; + }) +} + +export const replaceLinkPreviews = (text: string) => { + let parsed = text; + + const regex = /__LINK__.*?__LINK__/ig; + + parsed = parsed.replace(regex, (link) => { + const url = link.split('__LINK__')[1]; + + return `${url}`; + + }); + + return parsed; +} + +export const addlineBreaks = (text: string) => { + const regex = /(\r\n|\r|\n)/g; + + return text.replaceAll(regex, '
'); +}; + +export const highlightHashtags = (text: string) => { + const regex = /(^|\s)(#[a-z\d-]+)/ig; + + return text.replace(regex, "$1$2"); +}; + +export const parseNote1 = (content: string) => urlify(addlineBreaks(content)); +export const parseNote2 = (content: string) => urlify(addlineBreaks(content), true); +export const parseNote3 = (content: string) => urlify(addlineBreaks(content), false, false, true); + +type ReplyTo = { e?: string, p?: string }; +type NostrEvent = { content: string, kind: number, tags: string[][], created_at: number }; + +export const sendLike = async (note: PrimalNote, relays: Relay[]) => { + const event = { + content: '+', + kind: Kind.Reaction, + tags: [ + ['e', note.post.id], + ['p', note.post.pubkey], + ], + created_at: Math.floor((new Date()).getTime() / 1000), + }; + + return await sendEvent(event, relays); + +} + +export const sendRepost = async (note: PrimalNote, relays: Relay[]) => { + const event = { + content: JSON.stringify(note.msg), + kind: Kind.Repost, + tags: [ + ['e', note.post.id], + ['p', note.user.pubkey], + ], + created_at: Math.floor((new Date()).getTime() / 1000), + }; + + return await sendEvent(event, relays); +} + + +export const sendNote = async (text: string, relays: Relay[], tags: string[][]) => { + const event = { + content: text, + kind: Kind.Text, + tags, + created_at: Math.floor((new Date()).getTime() / 1000), + }; + + return await sendEvent(event, relays); +} + +export const sendContacts = async (contacts: string[], date: number, content: string, relays: Relay[]) => { + const event = { + content, + kind: Kind.Contacts, + tags: contacts.map(c => ['p', c]), + created_at: date, + }; + + return await sendEvent(event, relays); +}; + +export const sendEvent = async (event: NostrEvent, relays: Relay[]) => { + const win = window as NostrWindow; + const nostr = win.nostr; + + if (nostr === undefined) { + return false; + } + + const signedNote = await nostr.signEvent(event); + + return new Promise((resolve) => { + const numberOfRelays = relays.length; + let failed = 0; + + relays.forEach(relay => { + try { + let pub = relay.publish(signedNote); + + pub.on('ok', () => { + console.log(`${relay.url} has accepted our event`); + resolve(true); + }); + + pub.on('failed', (reason: any) => { + console.log(`failed to publish to ${relay.url}: ${reason}`) + failed += 1; + if (failed >= numberOfRelays) { + resolve(false); + } + }); + } catch (e) { + console.log('Failed sending note: ', e); + failed += 1; + if (failed >= numberOfRelays) { + resolve(false); + } + } + }); + + }); +} diff --git a/src/lib/notifications.ts b/src/lib/notifications.ts new file mode 100644 index 0000000..70be2ba --- /dev/null +++ b/src/lib/notifications.ts @@ -0,0 +1,127 @@ +import { Kind } from "../constants"; +import { sendMessage } from "../sockets"; +import { NostrWindow } from "../types/primal"; + +export const getNotifications = ( + user_pubkey: string | undefined, + pubkey: string | undefined, + subid: string, + since = 0, + limit = 1000, +) => { + if (!pubkey) { + return; + } + + let payload: { pubkey: string, limit: number, since: number, user_pubkey?: string } = { pubkey, limit, since }; + + if (user_pubkey) { + payload.user_pubkey = user_pubkey; + } + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["get_notifications", payload]}, + ])); +}; + +export const getOldNotifications = ( + user_pubkey: string | undefined, + pubkey: string | undefined, + subid: string, + until = 0, + limit = 20, +) => { + if (!pubkey) { + return; + } + + let payload: { pubkey: string, limit: number, until: number, user_pubkey?: string } = { pubkey, limit, until }; + + if (user_pubkey) { + payload.user_pubkey = user_pubkey; + } + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["get_notifications", payload]}, + ])); +}; + +export const getLastSeen = (pubkey: string | undefined, subid: string) => { + if (!pubkey) { + return; + } + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["get_notifications_seen", { pubkey }]}, + ])); + +}; + +export const setLastSeen = async ( + subid: string, + timestamp: number, +) => { + const win = window as NostrWindow; + const nostr = win.nostr; + + if (nostr === undefined) { + return false; + } + + const event = { + content: '{ "description": "update notifications last seen timestamp"}', + kind: Kind.Settings, + tags: [], + created_at: timestamp, + }; + + const signedNote = await nostr.signEvent(event); + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["set_notifications_seen", { + event_from_user: signedNote, + }]}, + ])); + +}; + +export const subscribeToNotificationStats = (pubkey: string, subid: string) => { + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["notification_counts", { pubkey, subid, }]}, + ])); +} + +export const truncateNumber = (amount: number, from?: 1 | 2 | 3 | 4) => { + const t = 1_000; + const s = from || 1; + + const l = Math.pow(t, s); + + if (amount < l) { + return amount.toLocaleString(); + } + + if (amount < Math.pow(t, 2)) { + return `${Math.floor(amount / t).toLocaleString()}K`; + } + + if (amount < Math.pow(t, 3)) { + return `${Math.floor(amount / Math.pow(t, 2)).toLocaleString()}M` + } + + if (amount < Math.pow(t, 4)) { + return `${Math.floor(amount / Math.pow(t, 3)).toLocaleString()}B` + } + + return `1T+`; +}; diff --git a/src/lib/profile.ts b/src/lib/profile.ts new file mode 100644 index 0000000..3fac2a2 --- /dev/null +++ b/src/lib/profile.ts @@ -0,0 +1,123 @@ +import { Event, Relay } from "nostr-tools"; +import { Kind, minKnownProfiles, noKey } from "../constants"; +import { sendMessage } from "../sockets"; +import { NostrWindow, VanityProfiles } from "../types/primal"; +import { getStorage } from "./localStore"; + +export const getUserProfiles = (pubkeys: string[], subid: string) => { + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["user_infos", { pubkeys }]}, + ])); +} + +export const getUserProfileInfo = (pubkey: string, subid: string) => { + if (pubkey === noKey) { + return + } + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["user_profile", { pubkey }]}, + ])); +} + +export const getProfileContactList = (pubkey: string, subid: string) => { + if (pubkey === noKey) { + return + } + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["contact_list", { pubkey }]}, + ])); +} + +export const getProfileScoredNotes = (pubkey: string, subid: string, limit = 5) => { + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["user_profile_scored_content", { pubkey, limit }]}, + ])); +} + +export const getTrendingUsers = (subid: string) => { + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["scored_users_24h"]}, + ])); +} + + +export const trimVerification = (address: string | undefined) => { + if (address === undefined) { + return ''; + } + + return address.split('@'); +} + +export const getLikes = (pubkey: string, relays: Relay[], callback: (likes: string[]) => void) => { + if (pubkey === noKey) { + return; + } + + const win = window as NostrWindow; + const nostr = win.nostr; + const storage = getStorage(pubkey); + + let likes = new Set(storage.likes); + + if (!nostr) { + callback(storage.likes); + return; + } + + // Request Reactions from all relays + try { + // const signedNote = await nostr.signEvent(event); + + relays.forEach(relay => { + + const sub = relay.sub([ + { + kinds: [Kind.Reaction], + authors: [pubkey], + }, + ]); + + sub.on('event', (event: Event) => { + const e = event.tags.find(t => t[0] === 'e'); + + e && e[1] && likes.add(e[1]); + }) + + sub.on('eose', () => { + const likeArray = Array.from(likes); + + callback(likeArray); + + sub.unsub(); + }) + }); + + } catch (e) { + console.log('Failed sending note: ', e); + } +}; + +export const fetchKnownProfiles: (vanityName: string) => Promise = async (vanityName: string) => { + try { + const name = vanityName.toLowerCase(); + const content = await fetch(`${window.location.origin}/.well-known/nostr.json?name=${name}`); + + return await content.json(); + } catch (e) { + console.log('Failed to fetch known users: ', e); + + return { ...minKnownProfiles }; + } +}; diff --git a/src/lib/relays.ts b/src/lib/relays.ts new file mode 100644 index 0000000..6249bf8 --- /dev/null +++ b/src/lib/relays.ts @@ -0,0 +1,66 @@ +import { relayInit, Relay } from "nostr-tools"; +import { relayConnectingTimeout } from "../constants"; +import { NostrRelays } from "../types/primal"; + +const logError = (relay: Relay, e: any, timedOut?: boolean) => { + const message = timedOut ? + 'timed-out connecting to relay: ' : + 'error connecting to relay: '; + + console.log(message, relay.url, e); +}; + +export const closeRelays = async (relays: Relay[], success = () => {}, fail = () => {}) => { + try { + for (let i=0; i< relays.length; i++) { + await relays[i].close() + } + return success(); + } catch (e) { + return fail(); + } +}; + +const connectToRelay = (relay: Relay) => new Promise( + (resolve, reject) => { + const timeout = setTimeout(() => { + relay.close(); + logError(relay, null, true); + reject(); + }, relayConnectingTimeout); + + relay.connect() + .then(() => { + clearTimeout(timeout); + resolve(true); + }) + .catch((e) => { + logError(relay, e); + reject(); + }); + }, +); + +export const connectRelays = async ( + relaySettings: NostrRelays, + onConnect: (relays: Relay[]) => void, +) => { + + const urls = Object.keys(relaySettings); + const relays = urls.map(u => relayInit(u)); + const connected: Relay[] = []; + + for (let i=0; i < relays.length; i++) { + const relay = relays[i]; + + if (relay.status === WebSocket.CLOSED) { + try { + await connectToRelay(relay); + connected.push(relay); + } catch(e){ + logError(relay, e); + }; + } + } + onConnect(connected); +}; diff --git a/src/lib/scroll.ts b/src/lib/scroll.ts new file mode 100644 index 0000000..5bc67d3 --- /dev/null +++ b/src/lib/scroll.ts @@ -0,0 +1,11 @@ +export const scrollWindowTo = (top: number = 0, smooth = false) => { + const behavior = smooth ? 'smooth' : 'instant'; + setTimeout(() => { + window.scrollTo({ + top, + left: 0, + // @ts-expect-error https://github.com/microsoft/TypeScript-DOM-lib-generator/issues/5 + behavior, + }); + }, 0); +}; diff --git a/src/lib/search.ts b/src/lib/search.ts new file mode 100644 index 0000000..8d11771 --- /dev/null +++ b/src/lib/search.ts @@ -0,0 +1,47 @@ +import { sendMessage } from "../sockets"; +import { sanitize } from "./notes"; + +type SearchPayload = { query: string, limit: number, pubkey?: string, since?: number, until?: number }; + +export const cleanQuery = (query: string) => { + return sanitize(query); +} + + +export const searchUsers = (pubkey: string | undefined, subid: string, query: string, limit = 10) => { + + let payload: SearchPayload = { query: cleanQuery(query), limit }; + + if (pubkey) { + payload.pubkey = pubkey; + } + + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["user_search", payload]}, + ])); +} + +export const searchContent = (subid: string, query: string, limit = 100) => { + + let payload: SearchPayload = { query: cleanQuery(query), limit }; + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["search", payload]}, + ])); +} + +export const searchFutureContent = (subid: string, query: string, since:number, limit = 100) => { + + let payload: SearchPayload = { query: cleanQuery(query), limit, since }; + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["search", payload]}, + ])); +} diff --git a/src/lib/settings.ts b/src/lib/settings.ts new file mode 100644 index 0000000..a8a7ed0 --- /dev/null +++ b/src/lib/settings.ts @@ -0,0 +1,68 @@ +import { Kind } from "../constants"; +import { sendMessage } from "../sockets"; +import { NostrWindow, PrimalFeed } from "../types/primal"; + +type PrimalSettings = { + theme: string, + feeds: PrimalFeed[], + description?: string, +}; + +export const sendSettings = async (settings: PrimalSettings, subid: string) => { + const win = window as NostrWindow; + const nostr = win.nostr; + + if (nostr === undefined) { + return false; + } + + const content = { description: 'Sync app settings', ...settings }; + + const event = { + content: JSON.stringify(content), + kind: Kind.Settings, + tags: [["d", "Primal-Web App"]], + created_at: Math.floor((new Date()).getTime() / 1000), + }; + + const signedNote = await nostr.signEvent(event); + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["set_app_settings", { settings_event: signedNote }]}, + ])) +}; + +export const getSettings = async (pubkey: string | undefined, subid: string) => { + const win = window as NostrWindow; + const nostr = win.nostr; + + if (nostr === undefined || !pubkey) { + return false; + } + + const event = { + content: '{ "description": "Sync app settings" }', + kind: Kind.Settings, + tags: [["d", "Primal-Web App"]], + created_at: Math.floor((new Date()).getTime() / 1000), + }; + + const signedNote = await nostr.signEvent(event); + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["get_app_settings", { event_from_user: signedNote }]}, + ])) +}; + +export const getDefaultSettings = async (subid: string) => { + + sendMessage(JSON.stringify([ + "REQ", + subid, + {cache: ["get_default_app_settings", { client: "Primal-Web App" }]}, + ])) +}; diff --git a/src/lib/stats.ts b/src/lib/stats.ts new file mode 100644 index 0000000..41424d7 --- /dev/null +++ b/src/lib/stats.ts @@ -0,0 +1,47 @@ +import { sendMessage } from '../sockets'; + +export const startListeningForNostrStats = (subId: string) => { + sendMessage(JSON.stringify([ + "REQ", + `netstats_${subId}`, + {cache: ["net_stats"]}, + ])); +}; + +export const stopListeningForNostrStats = (subId: string) => { + sendMessage(JSON.stringify([ + "CLOSE", + `netstats_${subId}`, + ])); +}; + +export const getLegendStats = (pubkey: string | undefined, subId: string) => { + pubkey && sendMessage(JSON.stringify([ + "REQ", + `legendstats_${subId}`, + {"cache":["explore_legend_counts",{ pubkey }]}, + ])); +} + +export const humanizeNumber = (number: number, veryShort = false) => { + + const bottomLimit = veryShort ? 1000 : 10000; + + if (number < bottomLimit) { + return number.toLocaleString(); + } + + if (number < 100000) { + return `${parseFloat((number/1000).toFixed(1))} k`; + } + + if (number < 1000000) { + return `${Math.floor(number/1000)} k`; + } + + if (number < 100000000) { + return `${parseFloat((number/1000000).toFixed(1))} m`; + } + + return `${Math.floor(number/1000000)} m`; +}; diff --git a/src/lib/textArea.ts b/src/lib/textArea.ts new file mode 100644 index 0000000..a4cb454 --- /dev/null +++ b/src/lib/textArea.ts @@ -0,0 +1,136 @@ + // Taken and slightly adapted from https://github.com/component/textarea-caret-position + + // We'll copy the properties below into the mirror div. + // Note that some browsers, such as Firefox, do not concatenate properties + // into their shorthand (e.g. padding-top, padding-bottom etc. -> padding), + // so we have to list every single property explicitly. + const properties = [ + 'direction', // RTL support + 'boxSizing', + 'width', // on Chrome and IE, exclude the scrollbar, so the mirror div wraps exactly as the textarea does + 'height', + 'overflowX', + 'overflowY', // copy the scrollbar for IE + + 'borderTopWidth', + 'borderRightWidth', + 'borderBottomWidth', + 'borderLeftWidth', + 'borderStyle', + + 'paddingTop', + 'paddingRight', + 'paddingBottom', + 'paddingLeft', + + // https://developer.mozilla.org/en-US/docs/Web/CSS/font + 'fontStyle', + 'fontVariant', + 'fontWeight', + 'fontStretch', + 'fontSize', + 'fontSizeAdjust', + 'lineHeight', + 'fontFamily', + + 'textAlign', + 'textTransform', + 'textIndent', + 'textDecoration', // might not make a difference, but better be safe + + 'letterSpacing', + 'wordSpacing', + + 'tabSize', + 'MozTabSize' + + ]; + + var isBrowser = (typeof window !== 'undefined'); + // @ts-ignore + var isFirefox = (isBrowser && window.mozInnerScreenX != null); + + export const getCaretCoordinates = (element: HTMLTextAreaElement | HTMLInputElement, position: number) => { + if (!isBrowser) { + throw new Error('textarea-caret-position#getCaretCoordinates should only be called in a browser'); + } + + // The mirror div will replicate the textarea's style + var div = document.createElement('div'); + div.id = 'input-textarea-caret-position-mirror-div'; + document.body.appendChild(div); + + var style = div.style; + // @ts-ignore + var computed = window.getComputedStyle ? window.getComputedStyle(element) : element.currentStyle; // currentStyle for IE < 9 + var isInput = element.nodeName === 'INPUT'; + + // Default textarea styles + style.whiteSpace = 'pre-wrap'; + if (!isInput) + style.wordWrap = 'break-word'; // only for textarea-s + + // Position off-screen + style.position = 'absolute'; // required to return coordinates properly + + // Transfer the element's properties to the div + properties.forEach(function (prop) { + if (isInput && prop === 'lineHeight') { + // Special case for s because text is rendered centered and line height may be != height + if (computed.boxSizing === "border-box") { + var height = parseInt(computed.height); + var outerHeight = + parseInt(computed.paddingTop) + + parseInt(computed.paddingBottom) + + parseInt(computed.borderTopWidth) + + parseInt(computed.borderBottomWidth); + var targetHeight = outerHeight + parseInt(computed.lineHeight); + if (height > targetHeight) { + style.lineHeight = height - outerHeight + "px"; + } else if (height === targetHeight) { + style.lineHeight = computed.lineHeight; + } else { + style.lineHeight = '0'; + } + } else { + style.lineHeight = computed.height; + } + } else { + // @ts-ignore + style[prop] = computed[prop]; + } + }); + + if (isFirefox) { + // Firefox lies about the overflow property for textareas: https://bugzilla.mozilla.org/show_bug.cgi?id=984275 + if (element.scrollHeight > parseInt(computed.height)) + style.overflowY = 'scroll'; + } else { + style.overflow = 'hidden'; // for Chrome to not render a scrollbar; IE keeps overflowY = 'scroll' + } + + div.textContent = element.value.substring(0, position); + // The second special handling for input type="text" vs textarea: + // spaces need to be replaced with non-breaking spaces - http://stackoverflow.com/a/13402035/1269037 + if (isInput) + div.textContent = div.textContent.replace(/\s/g, '\u00a0'); + + var span = document.createElement('span'); + // Wrapping must be replicated *exactly*, including when a long word gets + // onto the next line, with whitespace at the end of the line before (#7). + // The *only* reliable way to do that is to copy the *entire* rest of the + // textarea's content into the created at the caret position. + // For inputs, just '.' would be enough, but no need to bother. + span.textContent = element.value.substring(position) || '.'; // || because a completely empty faux span doesn't render at all + div.appendChild(span); + + var coordinates = { + top: span.offsetTop + parseInt(computed['borderTopWidth']), + left: span.offsetLeft + parseInt(computed['borderLeftWidth']), + height: parseInt(computed['lineHeight']) + }; + + document.body.removeChild(div); + + return coordinates; + } diff --git a/src/lib/zap.ts b/src/lib/zap.ts new file mode 100644 index 0000000..abd5405 --- /dev/null +++ b/src/lib/zap.ts @@ -0,0 +1,86 @@ +import { bech32 } from "@scure/base"; +import { nip57, Relay, utils } from "nostr-tools"; +import { NostrWindow, PrimalNote, PrimalUser } from "../types/primal"; + +export const zapNote = async (note: PrimalNote, sender: string | undefined, amount: number, comment = '', relays: Relay[]) => { + if (!sender) { + return false; + } + + const callback = await getZapEndpoint(note.user); + + if (!callback) { + return false; + } + + const sats = Math.round(amount * 1000); + + const zapReq = nip57.makeZapRequest({ + profile: note.post.pubkey, + event: note.msg.id, + amount: sats, + comment, + relays: relays.map(r => r.url) + }); + + const win = window as NostrWindow; + const nostr = win.nostr; + const webln = win.webln + + if (!nostr || !webln) { + return false; + } + + try { + const signedEvent = await nostr.signEvent(zapReq); + + const event = encodeURI(JSON.stringify(signedEvent)); + + const r2 = await (await fetch(`${callback}?amount=${sats}&nostr=${event}`)).json(); + const pr = r2.pr; + + await webln.enable(); + await webln.sendPayment(pr); + + return true; + } catch (e) { + return false; + } +} + +export const getZapEndpoint = async (user: PrimalUser): Promise => { + try { + let lnurl: string = '' + let {lud06, lud16} = user; + + if (lud16) { + let [name, domain] = lud16.split('@') + lnurl = `https://${domain}/.well-known/lnurlp/${name}` + } + else if (lud06) { + let {words} = bech32.decode(lud06, 1023) + let data = bech32.fromWords(words) + lnurl = utils.utf8Decoder.decode(data) + } + else { + return null; + } + + let res = await fetch(lnurl) + let body = await res.json() + + if (body.allowsNostr && body.nostrPubkey) { + return body.callback; + } + } catch (err) { + console.log('E: ', err); + return null; + /*-*/ + } + + return null; +} + +export const canUserReceiveZaps = (user: PrimalUser | undefined) => { + return !!user && (!!user.lud16 || !!user.lud06); +} diff --git a/src/logo.svg b/src/logo.svg new file mode 100644 index 0000000..025aa30 --- /dev/null +++ b/src/logo.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/src/pages/Downloads.module.scss b/src/pages/Downloads.module.scss new file mode 100644 index 0000000..e69de29 diff --git a/src/pages/Downloads.tsx b/src/pages/Downloads.tsx new file mode 100644 index 0000000..6a5b412 --- /dev/null +++ b/src/pages/Downloads.tsx @@ -0,0 +1,14 @@ +import { Component } from 'solid-js'; +import MissingPage from '../components/MissingPage/MissingPage'; + + +const Downloads: Component = () => { + + return ( + <> + + + ); +} + +export default Downloads; diff --git a/src/pages/Explore.module.scss b/src/pages/Explore.module.scss new file mode 100644 index 0000000..e937e94 --- /dev/null +++ b/src/pages/Explore.module.scss @@ -0,0 +1,175 @@ +.fullHeader { + display: grid; + height: 120px; + align-items: center; + justify-content: left; + position: relative; + margin-bottom: -3px; + position: relative; + + .exploreCaption { + font-weight: 300; + font-size: 32px; + line-height: 34px; + color: var(--brand-text); + } + + .addToFeed { + display: flex; + position: absolute; + bottom: 0px; + width: 100%; + height: 35px; + justify-content: flex-end; + align-items: flex-end; + + .noAdd { + display: flex; + align-items: center; + font-size: 16px; + line-height: 25px; + font-weight: 400; + color: var(--text-primary); + opacity: 0.6; + transition: opacity 0.4s; + } + + .addButton { + display: flex; + align-items: center; + margin: 0; + padding: 0; + border: none; + background-color: unset; + width: auto; + font-size: 16px; + line-height: 25px; + font-weight: 400; + color: var(--text-primary); + opacity: 0.6; + transition: opacity 0.4s; + + >span { + font-weight: 800; + margin-right: 5px; + } + + &:hover { + opacity: 1; + transition: opacity 0.4s; + } + + &:focus { + box-shadow: none; + } + } + } +} + +.exploreMenu { + display: grid; + grid-template-columns: 1fr 1fr 1fr 1fr; + grid-column-gap: 26px; + grid-row-gap: 26px; +} + +.exploreBox { + box-sizing: border-box; + + width: 140px; + height: 140px; + + border: 1px solid var(--text-tertiary-2); + border-radius: 9px; + + display: flex; + align-items: flex-end; + justify-content: center; + + text-decoration: none; + + >div { + display: flex; + width: 100%; + height: 100%; + flex-direction: column; + align-items: center; + + .firstLine { + text-align: center; + font-size: 18px; + font-weight: 300; + line-height: 22px; + color: var(--text-tertiary); + } + + .secondLine { + text-align: center; + font-size: 18px; + line-height: 22px; + font-weight: 700; + color: var(--text-primary); + } + + .exploreBoxIcon { + align-self: center; + justify-self: center; + margin-top: 12px; + margin-bottom: 12px; + } + } +} + +@mixin tableCell { + display: table-cell; + padding: 0px 12px; + margin: 12px 0px; + border-bottom: solid 1px var(--subtile-devider); +} + +.statsLegend { + display: table; + width: 100%; + margin-top: 53px; + + + .legendDetails { + display: table-row; + + .legendIcon { + @include tableCell; + width: 32px; + padding: 0px; + >img { + width: 32px; + height: 32px; + margin: 12px 0px; + } + } + + .legendName { + @include tableCell; + color: var(--text-primary); + font-size: 18px; + line-height: 22px; + font-weight: 700; + } + + .legendNumber { + @include tableCell; + color: var(--text-primary); + font-size: 24px; + line-height: 36px; + font-weight: 300; + + } + + .legendDescription { + @include tableCell; + color: var(--text-secondary); + font-size: 16px; + line-height: 22px; + font-weight: 300; + } + } +} diff --git a/src/pages/Explore.tsx b/src/pages/Explore.tsx new file mode 100644 index 0000000..8e6d5c7 --- /dev/null +++ b/src/pages/Explore.tsx @@ -0,0 +1,131 @@ +import { Component, Show } from 'solid-js'; +import styles from './Explore.module.scss'; +import ExploreMenu from './ExploreMenu'; +import Feed from './Feed'; +import { useParams } from '@solidjs/router'; +import Branding from '../components/Branding/Branding'; +import PageNav from '../components/PageNav/PageNav'; +import { scopeLabels, timeframeLabels } from '../constants'; +import ExploreSidebar from '../components/ExploreSidebar/ExploreSidebar'; +import { useToastContext } from '../components/Toaster/Toaster'; +import { useSettingsContext } from '../contexts/SettingsContext'; +import StickySidebar from '../components/StickySidebar/StickySidebar'; +import Wormhole from '../components/Wormhole/Wormhole'; +import { toast as t, explore as tExplore, actions as tAction } from '../translations'; +import { useIntl } from '@cookbook/solid-intl'; +import Search from '../components/Search/Search'; + + +const scopes = ['follows', 'tribe', 'network', 'global']; +const timeframes = ['latest', 'trending', 'popular', 'mostzapped']; + +const titleCase = (text: string) => { + return text[0].toUpperCase() + text.slice(1).toLowerCase(); +} + +const Explore: Component = () => { + + const settings = useSettingsContext(); + const toaster = useToastContext(); + const intl = useIntl(); + + const params = useParams(); + + const hasParams = () => { + if (!params.scope || !params.timeframe) { + return false; + } + + return scopes.includes(params.scope) && + timeframes.includes(params.timeframe); + + }; + + const hasFeedAtHome = () => { + const hex = `${params.scope};${params.timeframe}`; + + return !!settings?.availableFeeds.find(f => f.hex === hex); + }; + + const addToHomeFeed = () => { + const hex = `${params.scope};${params.timeframe}`; + const name = titleCase(`${timeframeLabels[params.timeframe]}, ${scopeLabels[params.scope]}`); + const feed = { name, hex }; + + settings?.actions.addAvailableFeed(feed); + + toaster?.sendSuccess(intl.formatMessage( + t.addFeedToHomeSuccess, + { name }, + )); + }; + + return ( + <> + + } + > + + + + + + + + +
+ + {intl.formatMessage(tExplore.genericCaption)} +
} + > +
+ {intl.formatMessage( + tExplore.title, + { + timeframe: timeframeLabels[params.timeframe], + scope: scopeLabels[params.scope], + }, + )} +
+
+ + {intl.formatMessage(tAction.disabledAddFeedToHome)} +
+ } + > + + + + + + + + + + + } + > + + + + ) +} + +export default Explore; diff --git a/src/pages/ExploreMenu.module.scss b/src/pages/ExploreMenu.module.scss new file mode 100644 index 0000000..3d5b60f --- /dev/null +++ b/src/pages/ExploreMenu.module.scss @@ -0,0 +1,314 @@ +.fullHeader { + display: grid; + height: 120px; + align-items: center; + justify-content: left; + + >div { + font-weight: 300; + font-size: 32px; + line-height: 34px; + color: var(--brand-text); + } +} + +.statsHolder { + margin-bottom: 36px; +} + +.exploreMenu { + display: grid; + grid-template-columns: 1fr 1fr 1fr 1fr; + grid-column-gap: 26px; + grid-row-gap: 26px; +} + +.exploreBox { + box-sizing: border-box; + + width: 140px; + height: 140px; + + border: 1px solid var(--text-tertiary-2); + border-radius: 9px; + + display: flex; + align-items: flex-end; + justify-content: center; + + text-decoration: none; + + >div { + display: flex; + width: 100%; + height: 100%; + flex-direction: column; + align-items: center; + + .firstLine { + text-align: center; + font-size: 18px; + font-weight: 300; + line-height: 22px; + color: var(--text-tertiary); + } + + .secondLine { + text-align: center; + font-size: 18px; + line-height: 22px; + font-weight: 700; + color: var(--text-primary); + } + + .exploreBoxIcon { + align-self: center; + justify-self: center; + margin-top: 12px; + margin-bottom: 12px; + } + } +} + +@mixin tableCell { + display: table-cell; + padding: 0px 12px; + border-bottom: solid 1px var(--text-tertiary-2); + vertical-align: middle; +} + +.statsLegend { + display: table; + width: 100%; + margin-top: 53px; + + + .legendDetails { + display: table-row; + + .legendIcon { + @include tableCell; + width: 32px; + padding: 0px; + >div { + width: 32px; + height: 32px; + margin: 12px 0px; + } + } + + .legendName { + @include tableCell; + color: var(--text-primary); + font-size: 18px; + line-height: 22px; + font-weight: 700; + } + + .legendNumber { + @include tableCell; + color: var(--text-primary); + font-size: 24px; + line-height: 36px; + font-weight: 300; + + } + + .legendDescription { + @include tableCell; + color: var(--text-secondary); + font-size: 16px; + line-height: 22px; + font-weight: 300; + } + } +} + +@mixin exploreCategoryIcon { + width: 56px; + height: 56px; + background-color: var(--background-site); +} + +.global_trending_icon { + @include exploreCategoryIcon; + background-image: var(--icon-global-trending); + background-size: contain; + background-repeat: no-repeat; +} + +.global_latest_icon { + @include exploreCategoryIcon; + background-image: var(--icon-network-latest); + background-size: contain; + background-repeat: no-repeat; +} + +.global_popular_icon { + @include exploreCategoryIcon; + background-image: var(--icon-network-popular); + background-size: contain; + background-repeat: no-repeat; +} + + +.network_trending_icon { + @include exploreCategoryIcon; + background-image: var(--icon-network-trending); + background-size: contain; + background-repeat: no-repeat; +} + +.network_latest_icon { + @include exploreCategoryIcon; + background-image: var(--icon-network-latest); + background-size: contain; + background-repeat: no-repeat; +} + +.network_popular_icon { + @include exploreCategoryIcon; + background-image: var(--icon-network-popular); + background-size: contain; + background-repeat: no-repeat; +} + + +.tribe_trending_icon { + @include exploreCategoryIcon; + background-image: var(--icon-tribe-trending); + background-size: contain; + background-repeat: no-repeat; +} + +.tribe_latest_icon { + @include exploreCategoryIcon; + background-image: var(--icon-tribe-latest); + background-size: contain; + background-repeat: no-repeat; +} + +.tribe_popular_icon { + @include exploreCategoryIcon; + background-image: var(--icon-tribe-popular); + background-size: contain; + background-repeat: no-repeat; +} + + +.follows_trending_icon { + @include exploreCategoryIcon; + background-image: var(--icon-follows-trending); + background-size: contain; + background-repeat: no-repeat; +} + +.follows_latest_icon { + @include exploreCategoryIcon; + background-image: var(--icon-follows-latest);; + background-size: contain; + background-repeat: no-repeat; +} + +.follows_popular_icon { + @include exploreCategoryIcon; + background-image: var(--icon-follows-popular);; + background-size: contain; + background-repeat: no-repeat; +} + + +.followsIcon { + @include exploreCategoryIcon; + background-image: var(--icon-follows); + background-size: contain; + background-repeat: no-repeat; +} + +.tribeIcon { + @include exploreCategoryIcon; + background-image: var(--icon-tribe); + background-size: contain; + background-repeat: no-repeat; +} + +.networkIcon { + @include exploreCategoryIcon; + background-image: var(--icon-network); + background-size: contain; + background-repeat: no-repeat; +} + +.globalIcon { + @include exploreCategoryIcon; + background-image: var(--icon-global); + background-size: contain; + background-repeat: no-repeat; +} + +@media only screen and (max-width: 720px) { + .exploreMenu { + width: 100%; + display: grid; + grid-template-columns: 1fr 1fr 1fr; + grid-template-rows: 1fr 1fr 1fr 1fr; + grid-column-gap: 6px; + grid-row-gap: 26px; + grid-template-areas: + "followsLatest followsTrending followsPopular" + "tribeLatest tribeTrending tribePopular" + "networkLatest networkTrending networkPopular" + "globalLatest globalTrending globalPopular"; + } + + .global_trending_box { + grid-area: globalTrending; + } + + .global_latest_box { + grid-area: globalLatest; + } + + .global_popular_box { + grid-area: globalPopular; + } + + + .network_trending_box { + grid-area: networkTrending; + } + + .network_latest_box { + grid-area: networkLatest; + } + + .network_popular_box { + grid-area: networkPopular; + } + + + .tribe_trending_box { + grid-area: tribeTrending; + } + + .tribe_latest_box { + grid-area: tribeLatest; + } + + .tribe_popular_box { + grid-area: tribePopular; + } + + + .follows_trending_box { + grid-area: followsTrending; + } + + .follows_latest_box { + grid-area: followsLatest; + } + + .follows_popular_box { + grid-area: followsPopular; + } +} diff --git a/src/pages/ExploreMenu.tsx b/src/pages/ExploreMenu.tsx new file mode 100644 index 0000000..5f7f4da --- /dev/null +++ b/src/pages/ExploreMenu.tsx @@ -0,0 +1,45 @@ +import { Component, createEffect, onCleanup } from 'solid-js'; +import { isConnected } from '../sockets'; +import styles from './ExploreMenu.module.scss'; + +import NostrStats from '../components/NostrStats/NostrStats'; +import ExploreMenuItem from '../components/ExploreMenuItem/ExploreMenuItem'; +import { initialExploreData, useExploreContext } from '../contexts/ExploreContext'; +import { useAccountContext } from '../contexts/AccountContext'; + + +const ExploreMenu: Component = () => { + + const explore = useExploreContext(); + const account = useAccountContext(); + + const legend = () => explore?.legend || { ...initialExploreData.legend }; + const stats = () => explore?.stats || { ...initialExploreData.stats }; + + createEffect(() => { + if (isConnected()) { + explore?.actions.fetchLegendStats(account?.publicKey); + explore?.actions.openNetStatsStream(); + } + }); + + onCleanup(() => { + explore?.actions.closeNetStatsStream(); + }); + + return ( + <> +
+ +
+ + + + + + + + ) +} + +export default ExploreMenu; diff --git a/src/pages/Feed.module.scss b/src/pages/Feed.module.scss new file mode 100644 index 0000000..485b84c --- /dev/null +++ b/src/pages/Feed.module.scss @@ -0,0 +1,3 @@ +.feedContent { + position: relative; +} diff --git a/src/pages/Feed.tsx b/src/pages/Feed.tsx new file mode 100644 index 0000000..d858eb7 --- /dev/null +++ b/src/pages/Feed.tsx @@ -0,0 +1,39 @@ +import { Component, createEffect, For, Show } from 'solid-js'; +import styles from './Feed.module.scss'; +import { useParams } from '@solidjs/router'; +import Note from '../components/Note/Note'; +import Loader from '../components/Loader/Loader'; +import { useExploreContext } from '../contexts/ExploreContext'; +import Paginator from '../components/Paginator/Paginator'; + +const Feed: Component<{ scope: string, timeframe: string}> = () => { + + const explore = useExploreContext(); + + const params = useParams(); + + createEffect(() => { + if (params.scope && params.timeframe) { + explore?.actions.clearNotes(); + explore?.actions.fetchNotes( + `${params.scope};${params.timeframe}`, + ); + } + }); + + return ( +
+ 0} + fallback={} + > + + {(note) => } + + + +
+ ) +} + +export default Feed; diff --git a/src/pages/Help.module.scss b/src/pages/Help.module.scss new file mode 100644 index 0000000..e69de29 diff --git a/src/pages/Help.tsx b/src/pages/Help.tsx new file mode 100644 index 0000000..f471b1d --- /dev/null +++ b/src/pages/Help.tsx @@ -0,0 +1,13 @@ +import { Component } from 'solid-js'; +import MissingPage from '../components/MissingPage/MissingPage'; + + +const Help: Component = () => { + return ( + <> + + + ); +} + +export default Help; diff --git a/src/pages/Home.module.scss b/src/pages/Home.module.scss new file mode 100644 index 0000000..9360baa --- /dev/null +++ b/src/pages/Home.module.scss @@ -0,0 +1,135 @@ +.homeContent { + position: relative; +} + +.paginate { + color: var(--text-tertiary-2); + position: absolute; + bottom: 1280px; + width: 640px; + height: 100px; +} + +.noContent, .endOfContent { + position: relative; + color: var(--text-secondary); + text-align: center; + margin-top: 80px; +} + +.normalCentralHeader { + display: block; +} + +.phoneCentralHeader { + display: none; +} + +.newContentItem { + width: 100%; + display: flex; + justify-content: center; + margin-top: 10px; + + >button { + display: flex; + align-items: center; + justify-content: center; + width: 100%; + height: 40px; + background: var(--background-card); + font-weight: 400; + font-size: 14px; + line-height: 18px; + border: none; + border-radius: 8px; + padding-block: 0; + padding-left: 2px; + margin: 0; + + .counter { + margin-left: 24px; + color: var(--text-secondary); + } + + &:hover { + background: var(--background-input); + .counter{ + color: var(--text-primary); + } + } + } +} + +.newContentNotification { + position: fixed; + top: 42px; + left: calc(calc(100vw - 1240px) / 2 + 176px + 32px); + width: 640px; + z-index: 20; + display: flex; + justify-content: center; + + >button { + display: flex; + align-items: center; + justify-content: center; + width: auto; + height: 40px; + background: var(--brand-gradient); + font-weight: 400; + font-size: 14px; + line-height: 18px; + border: none; + border-radius: 20px; + padding-block: 0; + padding-left: 4px; + margin: 0; + + .avatars { + display: flex; + align-items: center; + height: 40px; + .avatar { + border: solid 2px var(--text-primary); + border-radius: 50%; + width: 30px; + height: 30px; + transition: margin-right 0.2s; + margin-right: -9px; + } + } + + .counter { + margin-left: 24px; + } + } + +} +@media only screen and (max-width: 1300px) { + .newContentNotification { + left: calc(calc(100vw - 1032px) / 2 + 48px + 32px); + } +} + +@media only screen and (max-width: 1087px) { + .newContentNotification { + left: calc(calc(100vw - 720px) / 2 + 48px + 32px); + } +} + +@media only screen and (max-width: 720px) { + .newContentNotification { + left: 0; + width: 100%; + justify-content: center; + } + .normalCentralHeader { + display: none; + } + + .phoneCentralHeader { + display: block; + width: 100%; + } +} diff --git a/src/pages/Home.tsx b/src/pages/Home.tsx new file mode 100644 index 0000000..c2fcdf2 --- /dev/null +++ b/src/pages/Home.tsx @@ -0,0 +1,230 @@ +import { + Component, + createEffect, + createSignal, + For, + Match, + onCleanup, + onMount, + Show, + Switch +} from 'solid-js'; +import Note from '../components/Note/Note'; +import styles from './Home.module.scss'; +import HomeHeader from '../components/HomeHeader/HomeHeader'; +import Loader from '../components/Loader/Loader'; +import Paginator from '../components/Paginator/Paginator'; +import HomeSidebar from '../components/HomeSidebar/HomeSidebar'; +import Branding from '../components/Branding/Branding'; +import HomeHeaderPhone from '../components/HomeHeaderPhone/HomeHeaderPhone'; +import Wormhole from '../components/Wormhole/Wormhole'; +import { scrollWindowTo } from '../lib/scroll'; +import StickySidebar from '../components/StickySidebar/StickySidebar'; +import { useHomeContext } from '../contexts/HomeContext'; +import { useIntl } from '@cookbook/solid-intl'; +import { createStore } from 'solid-js/store'; +import { PrimalUser } from '../types/primal'; +import Avatar from '../components/Avatar/Avatar'; +import { userName } from '../stores/profile'; +import { useAccountContext } from '../contexts/AccountContext'; +import { feedNewPosts, placeholders } from '../translations'; +import Search from '../components/Search/Search'; + + +const Home: Component = () => { + + const context = useHomeContext(); + const account = useAccountContext(); + const intl = useIntl(); + + const isPageLoading = () => context?.isFetching; + + let checkNewNotesTimer: number = 0; + + const [hasNewPosts, setHasNewPosts] = createSignal(false); + const [newNotesCount, setNewNotesCount] = createSignal(0); + const [newPostAuthors, setNewPostAuthors] = createStore([]); + + + const newPostCount = () => newNotesCount() < 100 ? newNotesCount() : 100; + + + onMount(() => { + scrollWindowTo(context?.scrollTop); + }); + + createEffect(() => { + const hex = context?.selectedFeed?.hex; + + if (checkNewNotesTimer) { + clearInterval(checkNewNotesTimer); + setHasNewPosts(false); + setNewNotesCount(0); + setNewPostAuthors(() => []); + } + + const timeout = 25_000 + Math.random() * 10_000; + + checkNewNotesTimer = setInterval(() => { + context?.actions.checkForNewNotes(hex); + }, timeout); + + }); + + createEffect(() => { + const count = context?.future.notes.length || 0; + if (count === 0) { + return + } + + if (!hasNewPosts()) { + setHasNewPosts(true); + } + + if (newPostAuthors.length < 3) { + const users = context?.future.notes.map(note => note.user) || []; + + const uniqueUsers = users.reduce((acc, user) => { + const isDuplicate = acc.find(u => u.pubkey === user.pubkey); + return isDuplicate ? acc : [ ...acc, user ]; + }, []).slice(0, 3); + + setNewPostAuthors(() => [...uniqueUsers]); + } + + setNewNotesCount(count); + + + }); + + onCleanup(()=> { + clearInterval(checkNewNotesTimer); + }); + + const loadNewContent = () => { + if (newNotesCount() > 100) { + location.reload(); + return; + } + context?.actions.loadFutureContent(); + scrollWindowTo(0, true); + setHasNewPosts(false); + setNewNotesCount(0); + setNewPostAuthors(() => []); + } + + return ( +
+ + + + + + + + +
+ +
+ +
+ +
+ + 40) && + !account?.showNewNoteForm + }> +
+ +
+
+ + + + + + +
+ +
+
+ + 0} + > + + {note => } + + + + + +
+ +
+
+ +
+ {intl.formatMessage(placeholders.endOfFeed)} +
+
+ +
+ +
+
+
+ +
+ ) +} + +export default Home; diff --git a/src/pages/Messages.module.scss b/src/pages/Messages.module.scss new file mode 100644 index 0000000..22600b5 --- /dev/null +++ b/src/pages/Messages.module.scss @@ -0,0 +1,427 @@ +@mixin messageContent { + padding-block: 8px; + padding-inline: 16px; + margin-bottom: 8px; + width: fit-content; + max-width: calc(640px + 12px - 40px); +} + +@mixin thread($align-end) { + display: flex; + flex-direction: column; + margin-top: -16px; + + @if $align-end { + align-items: flex-end; + } @else { + align-items: flex-start; + } + + .avatar { + margin-bottom: 8px; + } + .threadMessages { + display: flex; + flex-direction: column-reverse; + align-items: flex-start; + max-width: calc(100% - 48px); + + .message { + @include messageContent(); + + @if $align-end { + align-self: flex-end; + text-align: left; + border-radius: 8px 0px 8px 8px; + } @else { + align-items: flex-start; + text-align: left; + border-radius: 0px 8px 8px 8px; + } + + font-weight: 400; + font-size: 16px; + line-height: 24px; + color: var(--text-primary); + background-color: var(--subtile-devider); + } + } + .threadTime { + color: var(--text-tertiary-2); + font-weight: 400; + font-size: 12px; + line-height: 16px; + } +} + +.fullHeader { + display: grid; + height: 120px; + align-items: center; + justify-content: left; + padding-top: 6px; + + >div { + font-weight: 300; + font-size: 32px; + line-height: 34px; + color: var(--brand-text); + text-transform: lowercase; + } +} + +.messagesContent { + position: relative; + + .sendersHeader { + height: 40px; + width: 294px; + background-color: var(--background-input); + border-radius: 8px; + display: flex; + justify-content: space-between; + align-items: center; + padding-inline: 12px; + margin-bottom: 8px; + .senderCategorySelector { + display: flex; + .categorySelector { + border: none; + outline: none; + padding: 0; + margin: 0; + background: none; + color: var(--text-tertiary); + font-weight: 400; + font-size: 14px; + line-height: 16px; + text-transform: uppercase; + + &:focus { + background: none; + box-shadow: none; + outline: none; + } + + &.highlight, &:hover { + color: var(--text-primary); + } + + } + .separator { + border-left: 1px solid var(--subtile-devider); + margin-inline: 8px; + } + } + + .markAsRead { + border: none; + outline: none; + padding: 0; + margin: 0; + background: none; + color: var(--accent-1); + font-weight: 400; + font-size: 13px; + line-height: 16px; + width: auto; + + &:focus { + background: none; + box-shadow: none; + outline: none; + } + } + } + + .sendersList { + display: flex; + flex-direction: column; + overflow-y: scroll; + width: 308px; + height: calc(100vh - 176px); + padding-right: 8px; + + + .senderItem { + position: relative; + display: flex; + height: 65px; + background-color: var(--background-card); + padding-inline: 15px; + padding-block: 12px; + border-radius: 8px; + border: none; + margin-bottom: 8px; + align-items: center; + + &:hover, &.selected { + background-color: var(--background-input); + } + + .senderInfo { + margin-left: 12px; + display: flex; + flex-direction: column; + font-size: 16px; + line-height: 16px; + .firstLine { + display: flex; + justify-content: flex-start; + .senderName { + color: var(--text-primary); + font-weight: 700; + } + .lastMessageTime { + color: var(--text-tertiary-2); + font-weight: 400; + margin-left: 2px; + + &::before { + content: "|"; + padding: 0px 2px; + } + } + } + + .secondLine { + text-align: left; + color: var(--text-tertiary); + font-weight: 400; + font-size: 14px; + line-height: 16px; + margin-top: 4px; + padding: 0; + max-width: 200px; + text-overflow: ellipsis; + white-space: nowrap; + overflow: hidden; + } + } + } + } + + .conversation { + position: absolute; + top: 0; + left: 320px; + background-color: var(--background-card ); + border-radius: 8px; + width: calc(640px + 12px); + height: calc(100vh - 128px); + z-index: var(--z-index-lifted); + overflow: hidden; + padding: 12px; + display: flex; + flex-direction: column-reverse; + + .messages { + width: 100%; + height: calc(100% - 44px); + margin-bottom: 12px; + overflow-y: scroll; + display: flex; + flex-direction: column-reverse; + padding-right: 8px; + position: relative; + + .myThread { + @include thread(true); + .threadMessages { + .message { + background-color: var(--subtile-devider); + } + } + + .myThread { + margin-bottom: 20px; + } + } + + .theirThread { + @include thread(false); + .threadMessages { + .message { + background-color: var(--background-input); + } + } + + .theirThread { + margin-bottom: 20px; + } + } + } + + .newMessage { + height: 32px; + width: 100%; + position: relative; + vertical-align: bottom; + display: table-cell; + + .textAreaBorder { + padding: 1px; + background: var(--brand-gradient); + border-radius: 8px; + width: 510px; + height: 34px; + box-sizing: border-box; + + textarea { + // border: 1px solid var(--brand-gradient-vertical); + background: var(--background-site); + border-radius: 8px; + color: var(--text-primary); + font-weight: 400; + font-size: 14px; + line-height: 20px; + height: 32px; + margin: 0; + margin-right: 12px; + padding-inline: 8px; + padding-block: 8px; + width: 508px; + max-height: none; + + &:focus { + box-shadow: none; + } + } + } + + .secondaryButton { + position: absolute; + top: 0; + right: 14px; + width: 80px; + height: 32px; + border: none; + border-radius: 6px; + padding: 1px; + font-size: 14px; + line-height: 20px; + font-weight: 700; + background: var(--brand-gradient-vertical); + color: var(--text-tertiary-2); + >div { + width: 100%; + height: 100%; + vertical-align: middle; + border-radius: 6px; + background-color: var(--background-card); + display: flex; + flex-direction: column; + align-items: center; + justify-content: center; + } + } + + .primaryButton { + position: absolute; + top: 0; + right: 14px; + width: 80px; + height: 32px; + border: none; + border-radius: 6px; + padding: 0px; + font-size: 14px; + line-height: 20px; + font-weight: 700; + background: var(--brand-gradient-vertical); + color: white; + >span { + opacity: 0.75; + } + } + } + } +} + +.senderBubble { + position: absolute; + text-align: center; + padding-top: 2px; + padding-inline: 4px; + top: 12px; + right: 12px; + min-width: 18px; + min-height: 18px; + border-radius: 8px; + font-weight: 500; + font-size: 12px; + line-height: 12px; + z-index: var(--z-index-lifted); + + background: var(--brand-gradient); + border: 1px solid var(--background-site); + + color: white; + text-shadow: 0.5px 0.5px 0px black; + + &.doubleSize { + right: -24px; + } + &.tripleSize { + right: -30px; + } +} + +.postLink { + text-decoration: none; + color: unset; + max-height: 650px; + overflow: hidden; +} + +.searchSuggestions { + width: 300px; + background-color: var(--background-site); + border: 1px solid var(--text-tertiary-2); + // box-shadow: 0px 4px 12px rgba(0, 0, 0, 0.8); + border-radius: 4px; + + position: absolute; + bottom: 0px; + left: 0px; + z-index: var(--z-index-header); +} + + +.emojiSuggestions { + position: absolute; + display: grid; + grid-template-columns: 50px 50px 50px 50px 50px 50px; + width: 322px; + max-height: 200px; + overflow-y: scroll; + padding: 4px; + background-color: var(--background-site); + border: 1px solid var(--text-tertiary-2); + // box-shadow: 0px 4px 12px rgba(0, 0, 0, 0.8); + border-radius: 8px; + + bottom: 32px; + left: 0px; + z-index: var(--z-index-floater); + + .emojiOption { + margin-bottom: 5px; + padding: 2px; + background: none; + font-size: 16px; + line-height: 20px; + font-weight: 400; + border: none; + display: flex; + justify-content: center; + align-items: center; + + &:hover, &.highlight { + background-color: var(--text-tertiary-2); + } + + &:focus { + outline: none; + border: none; + } + } +} diff --git a/src/pages/Messages.tsx b/src/pages/Messages.tsx new file mode 100644 index 0000000..9f47160 --- /dev/null +++ b/src/pages/Messages.tsx @@ -0,0 +1,1141 @@ +import { useIntl } from '@cookbook/solid-intl'; +import { nip19 } from 'nostr-tools'; +import { Component, createEffect, createSignal, For, onCleanup, onMount, Show } from 'solid-js'; +import Avatar from '../components/Avatar/Avatar'; +import { useAccountContext } from '../contexts/AccountContext'; +import { useMessagesContext } from '../contexts/MessagesContext'; +import { nip05Verification, truncateNpub, userName } from '../stores/profile'; +import { PrimalNote, PrimalUser } from '../types/primal'; +import { date } from '../lib/dates'; + +import styles from './Messages.module.scss'; +import EmbeddedNote from '../components/EmbeddedNote/EmbeddedNote'; +import { A, useNavigate, useParams } from '@solidjs/router'; +import { parseNote3 } from '../lib/notes'; +import { hexToNpub } from '../lib/keys'; +import Branding from '../components/Branding/Branding'; +import Wormhole from '../components/Wormhole/Wormhole'; +import Loader from '../components/Loader/Loader'; +import SearchOption from '../components/Search/SearchOption'; +import { debounce, isVisibleInContainer, uuidv4 } from '../utils'; +import { useSearchContext } from '../contexts/SearchContext'; +import { createStore } from 'solid-js/store'; +import { editMentionRegex } from '../constants'; +import Search from '../components/Search/Search'; +import { useProfileContext } from '../contexts/ProfileContext'; +import Paginator from '../components/Paginator/Paginator'; +import { getCaretCoordinates } from '../lib/textArea'; +import emojiSearch from '@jukben/emoji-search'; +import { + placeholders, + messages as tMessages, + actions as tActions, + search as tSearch, +} from '../translations'; + +type AutoSizedTextArea = HTMLTextAreaElement & { _baseScrollHeight: number }; + +let currentUrl = ''; + +type EmojiOption = { + keywords: string[], + char: string, + fitzpatrick_scale: boolean, + category: string, + name: string, +}; + +export const parseNoteLinks = (text: string, mentionedNotes: Record, mentionedUsers: Record, highlightOnly?: boolean) => { + + const regex = /\bnostr:((note|nevent)1\w+)\b|#\[(\d+)\]/g; + + return text.replace(regex, (url) => { + const [_, id] = url.split(':'); + + if (!id) { + return url; + } + + try { + const note = mentionedNotes[id]; + + const path = `/thread/${id}`; + + const link = highlightOnly ? + {url} : + note ? + + + : + {url}; + + // @ts-ignore + return link.outerHTML || url; + } catch (e) { + return `${url}`; + } + + }); + +}; + +export const parseNpubLinks = (text: string, mentionedUsers: Record, highlightOnly = false) => { + + const regex = /\bnostr:((npub|nprofile)1\w+)\b|#\[(\d+)\]/g; + + return text.replace(regex, (url) => { + const [_, id] = url.split(':'); + + if (!id) { + return url; + } + + try { + const profileId = nip19.decode(id).data as string | nip19.ProfilePointer; + + const hex = typeof profileId === 'string' ? profileId : profileId.pubkey; + const npub = hexToNpub(hex); + const path = `/profile/${npub}`; + + const user = mentionedUsers[hex]; + + let link = highlightOnly ? + @{truncateNpub(npub)} : + @{truncateNpub(npub)}; + + if (user) { + link = highlightOnly ? + @{userName(user)} : + @{userName(user)}; + } + + + // @ts-ignore + return link.outerHTML || url; + } catch (e) { + return `${url}`; + } + }); + +}; + +const emojiSearchLimit = 2; + +const Messages: Component = () => { + const instanceId = uuidv4(); + + const intl = useIntl(); + const messages = useMessagesContext(); + const account = useAccountContext(); + const profile = useProfileContext(); + + const navigate = useNavigate(); + + const params = useParams(); + + let conversationHolder: HTMLDivElement | undefined; + let newMessageInput: HTMLTextAreaElement | undefined; + let newMessageInputBorder: HTMLDivElement | undefined; + let newMessageWrapper: HTMLDivElement | undefined; + let sendersListElement: HTMLDivElement | undefined; + + let emojiOptions: HTMLDivElement | undefined; + + const [highlightedEmoji, setHighlightedEmoji] = createSignal(0); + const [isEmojiInput, setEmojiInput] = createSignal(false); + const [emojiQuery, setEmojiQuery] = createSignal(''); + const [emojiResults, setEmojiResults] = createStore([]); + let emojiCursorPosition = { top: 0, left: 0, height: 0 }; + + const senderNpub = () => { + if (!params.sender) { + return ''; + } + + if (params.sender.startsWith('npub')) { + return params.sender; + } + + return nip19.noteEncode(params.sender); + }; + + const orderedSenders = () => { + if (!messages || !messages.senders) { + return []; + } + const senders = messages.senders; + const counts = messages.messageCountPerSender; + + const ids = Object.keys(senders); + const latests = ids.map(id => ({ latest_at: counts[id]?.latest_at || null, id })); + + const ordered = latests.sort((a, b) => { + if (!a.latest_at) { + return -1; + } + + if (!b.latest_at) { + return 1; + } + + return b.latest_at - a.latest_at + }); + + return ordered.map(o => senders[o.id]); + }; + + const senderPubkey = () => { + if (!params.sender) { + return ''; + } + + let pubkey = params.sender; + + if (pubkey.startsWith('npub') || pubkey.startsWith('nevent')) { + const decoded = nip19.decode(pubkey); + + if (decoded.type === 'npub') { + pubkey = decoded.data; + } + + if (decoded.type === 'nevent') { + pubkey = decoded.data.id; + } + } + + return pubkey; + + } + + createEffect(() => { + if(params.sender && currentUrl !== params.sender) { + currentUrl = params.sender; + messages?.actions.selectSender(params.sender); + } + }); + + createEffect(() => { + if (messages?.selectedSender && + currentUrl !== messages?.selectedSender?.npub + ) { + navigate(`/messages/${messages?.selectedSender.npub}`); + return; + } + }); + + createEffect(() => { + if (params.sender || !messages?.senders) { + return; + } + + const senderIds = Object.keys(messages.senders); + senderIds.length > 0 && navigate(`/messages/${messages.senders[senderIds[0]].npub}`); + return; + + }); + + createEffect(() => { + const count = messages?.messageCount || 0; + + if (account?.isKeyLookupDone && account.hasPublicKey() && count === 0) { + messages?.actions.getMessagesPerSender(); + } + }); + + createEffect(() => { + const count = messages?.messageCount || 0; + + if (count > 0) { + messages?.actions.getMessagesPerSender(); + } + }) + + createEffect(() => { + if (messages?.isConversationLoaded) { + if (conversationHolder) { + conversationHolder.scrollTop = conversationHolder.scrollHeight; + } + + // messages.actions.resetConversationLoaded(); + } + }); + + const user = (pubkey: string) => { + return messages?.senders && messages.senders[pubkey]; + } + + const mgsFromSender = (sender: PrimalUser) => { + return messages?.messageCountPerSender[sender.pubkey]?.cnt || 0; + } + + const isSelectedSender = (senderId: string) => { + return senderNpub() === senderId || senderPubkey() === senderId; + }; + + const selectSender = (senderNpub: string) => { + messages?.actions.selectSender(senderNpub); + } + + const highlightHashtags = (text: string) => { + const regex = /(?:\s|^)#[^\s!@#$%^&*(),.?":{}|<>]+/ig; + + return text.replace(regex, (token) => { + const [space, term] = token.split('#'); + const embeded = ( + + {space} + #{term} + + ); + + // @ts-ignore + return embeded.outerHTML; + }); + } + + const parseMessage = (message: string) => { + if (!messages) { + return message; + } + return parseNoteLinks( + parseNpubLinks( + highlightHashtags( + parseNote3(message) + ), + messages?.referecedUsers, + ), + messages?.referecedNotes, + messages?.referecedUsers + ); + }; + + + const getScrollHeight = (elm: AutoSizedTextArea) => { + var savedValue = elm.value + elm.value = '' + elm._baseScrollHeight = elm.scrollHeight + elm.value = savedValue + } + + const [message, setMessage] = createSignal(''); + + const onExpandableTextareaInput = () => { + const maxHeight = 800; + + const elm = newMessageInput as AutoSizedTextArea; + + if(!elm || elm.nodeName !== 'TEXTAREA') { + return; + } + + const minRows = parseInt(elm.getAttribute('data-min-rows') || '0'); + + !elm._baseScrollHeight && getScrollHeight(elm); + + + if (elm.scrollHeight >= (maxHeight / 3)) { + return; + } + + elm.rows = minRows; + const rows = elm.value === '' ? 0 : Math.ceil((elm.scrollHeight - elm._baseScrollHeight) / 20); + + elm.rows = minRows + rows; + elm.style.height = `${32 + (20 * rows)}px`; + + if (newMessageWrapper) { + newMessageWrapper.style.height = `${32 + (20 * rows)}px`; + } + + if (newMessageInputBorder) { + newMessageInputBorder.style.height = `${34 + (20 * rows)}px`; + } + + // debounce(() => { + setMessage(elm.value) + // }, 300); + + } + + + const onKeyDown = (e: KeyboardEvent) => { + if (!newMessageInput || !newMessageWrapper) { + return false; + } + + const mentionSeparators = ['Enter', 'Space', 'Comma']; + + if (!isMentioning() && !isEmojiInput() && e.code === 'Enter' && !e.shiftKey) { + e.preventDefault(); + debounce(() => { + sendMessage(); + }, 300); + + return false; + } + + if (!isMentioning() && !isEmojiInput() && e.key === ':') { + emojiCursorPosition = getCaretCoordinates(newMessageInput, newMessageInput.selectionStart); + setEmojiInput(true); + return false; + } + + if (isEmojiInput()) { + + if (e.code === 'ArrowDown') { + e.preventDefault(); + setHighlightedEmoji(i => { + if (emojiResults.length === 0) { + return 0; + } + + return i < emojiResults.length - 7 ? i + 6 : 0; + }); + + const emojiHolder = document.getElementById(`${instanceId}-${highlightedEmoji()}`); + + if (emojiHolder && emojiOptions && !isVisibleInContainer(emojiHolder, emojiOptions)) { + emojiHolder.scrollIntoView({ block: 'end', behavior: 'smooth' }); + } + + return false; + } + + if (e.code === 'ArrowUp') { + e.preventDefault(); + setHighlightedEmoji(i => { + if (emojiResults.length === 0) { + return 0; + } + + return i >= 6 ? i - 6 : emojiResults.length - 1; + }); + + const emojiHolder = document.getElementById(`${instanceId}-${highlightedEmoji()}`); + + if (emojiHolder && emojiOptions && !isVisibleInContainer(emojiHolder, emojiOptions)) { + emojiHolder.scrollIntoView({ block: 'start', behavior: 'smooth' }); + } + + return false; + } + + if (e.code === 'ArrowRight') { + e.preventDefault(); + setHighlightedEmoji(i => { + if (emojiResults.length === 0) { + return 0; + } + + return i < emojiResults.length - 1 ? i + 1 : 0; + }); + + const emojiHolder = document.getElementById(`${instanceId}-${highlightedEmoji()}`); + + if (emojiHolder && emojiOptions && !isVisibleInContainer(emojiHolder, emojiOptions)) { + emojiHolder.scrollIntoView({ block: 'end', behavior: 'smooth' }); + } + + return false; + } + + if (e.code === 'ArrowLeft') { + e.preventDefault(); + setHighlightedEmoji(i => { + if (emojiResults.length === 0) { + return 0; + } + + return i > 0 ? i - 1 : emojiResults.length - 1; + }); + + const emojiHolder = document.getElementById(`${instanceId}-${highlightedEmoji()}`); + + if (emojiHolder && emojiOptions && !isVisibleInContainer(emojiHolder, emojiOptions)) { + emojiHolder.scrollIntoView({ block: 'start', behavior: 'smooth' }); + } + + return false; + } + + if (mentionSeparators.includes(e.code)) { + if (emojiQuery().trim().length === 0) { + setEmojiInput(false); + return false; + } + e.preventDefault(); + selectEmoji(emojiResults[highlightedEmoji()]); + setHighlightedEmoji(0); + return false; + } + + const cursor = newMessageInput.selectionStart; + const lastEmojiTrigger = newMessageInput.value.slice(0, cursor).lastIndexOf(':'); + + if (e.code === 'Backspace') { + setEmojiQuery(emojiQuery().slice(0, -1)); + + if (lastEmojiTrigger < 0 || cursor - lastEmojiTrigger <= 1) { + setEmojiInput(false); + return false; + } + } else { + setEmojiQuery(q => q + e.key); + return false; + } + + // if (emojiQuery().length === 0) { + // setEmojiInput(false); + // return false; + // } + + return false; + } + + if (!isMentioning() && e.key === '@') { + mentionCursorPosition = getCaretCoordinates(newMessageInput, newMessageInput.selectionStart); + setPreQuery(''); + setQuery(''); + setMentioning(true); + return false; + } + + if (!isMentioning() && e.code === 'Backspace' && newMessageInput) { + let cursor = newMessageInput.selectionStart; + const textSoFar = newMessageInput.value.slice(0, cursor); + const lastWord = textSoFar.split(/[\s,;\n\r]/).pop(); + + if (lastWord?.startsWith('@`')) { + const index = textSoFar.lastIndexOf(lastWord); + + const newText = textSoFar.slice(0, index) + newMessageInput.value.slice(cursor); + + setMessage(newText); + newMessageInput.value = newText; + + newMessageInput.selectionEnd = index; + } + } + + if (isMentioning()) { + + if (e.code === 'ArrowDown') { + e.preventDefault(); + setHighlightedUser(i => { + if (!search?.users || search.users.length === 0) { + return 0; + } + + return i < search.users.length - 1 ? i + 1 : 0; + }); + return false; + } + + if (e.code === 'ArrowUp') { + e.preventDefault(); + setHighlightedUser(i => { + if (!search?.users || search.users.length === 0) { + return 0; + } + + return i > 0 ? i - 1 : search.users.length - 1; + }); + return false; + } + + if (mentionSeparators.includes(e.code)) { + if (preQuery().trim().length === 0) { + setMentioning(false); + return false; + } + e.preventDefault(); + search?.users && selectUser(search.users[highlightedUser()]) + setMentioning(false); + return false; + } + + const cursor = newMessageInput.selectionStart; + const lastMentionTrigger = newMessageInput.value.slice(0, cursor).lastIndexOf('@'); + + if (e.code === 'Backspace') { + setPreQuery(preQuery().slice(0, -1)); + + if (lastMentionTrigger < 0 || cursor - lastMentionTrigger <= 1) { + setMentioning(false); + return false; + } + } else { + setPreQuery(q => q + e.key); + return false + } + + // if (preQuery().length === 0) { + // setMentioning(false); + // return false; + // } + + return false; + } + + return true; + }; + + // const onKeyDown = (e: KeyboardEvent) => { + // if (!newMessageInput) { + // return false; + // } + + // if (e.code === 'Enter' && !e.shiftKey) { + // e.preventDefault(); + // debounce(() => { + // sendMessage(); + // }, 300); + + // return false; + // } + + // if (!isMentioning() && !isEmojiInput() && e.key === ':') { + // emojiCursorPosition = getCaretCoordinates(newMessageInput, newMessageInput.selectionStart); + // setEmojiInput(true); + // return false; + // } + // }; + + onMount(() => { + newMessageWrapper?.addEventListener('input', () => onExpandableTextareaInput()); + newMessageInput && newMessageInput.addEventListener('keydown', onKeyDown); + }); + + onCleanup(() => { + newMessageWrapper?.removeEventListener('input', () => onExpandableTextareaInput()); + newMessageInput && newMessageInput.removeEventListener('keydown', onKeyDown); + }); + + const sendMessage = async () => { + if (!messages?.selectedSender || + !newMessageInput || + !newMessageInputBorder || + !newMessageWrapper) { + return; + } + + const text = message().trim(); + + if (text.length === 0) { + return; + } + setMessage(''); + + const content = prepareMessageForSending(text); + + const msg = { + id: `N_M_${messages.messages.length}`, + sender: account?.publicKey || '', + content, + created_at: Math.floor((new Date()).getTime() / 1000), + }; + + const success = await messages?.actions.sendMessage(messages.selectedSender, msg); + + if (success) { + newMessageInput.value = ''; + newMessageInput.style.height = '32px'; + newMessageInputBorder.style.height = '34px'; + newMessageWrapper.style.height = '32px'; + + setTimeout(() => { + const element = document.querySelector(`[data-user="${messages?.selectedSender?.pubkey}"]`); + + if (element && sendersListElement && !isVisibleInContainer(element, sendersListElement)) { + element.scrollIntoView(); + } + }, 100); + } + }; + + const [inputFocused, setInputFocused] = createSignal(false); + + const markAllAsRead = () => { + messages?.actions.resetAllMessages(); + }; + + const sendButtonClass = () => { + return inputFocused() && message().trim().length > 0 ? styles.primaryButton : styles.secondaryButton; + }; + + const addUserToSenders = (user: PrimalUser | string) => { + if (typeof user === 'string') { + return; + } + + messages?.actions.addSender(user); + } + +// MENTIONING + + const search = useSearchContext(); + + const [isMentioning, setMentioning] = createSignal(false); + const [preQuery, setPreQuery] = createSignal(''); + const [query, setQuery] = createSignal(''); + + const [highlightedUser, setHighlightedUser] = createSignal(0); + let mentionCursorPosition = { top: 0, left: 0, height: 0 }; + + let mentionOptions: HTMLDivElement | undefined; + + const prepareMessageForSending = (text: string) => { + + return text.replace(editMentionRegex, (url) => { + + const [_, name] = url.split('\`'); + const user = userRefs[name]; + + // @ts-ignore + return ` nostr:${user.npub}`; + }) + } + + createEffect(() => { + const preQ = preQuery(); + + debounce(() => { + setQuery(() => preQ) + }, 500); + }) + + createEffect(() => { + if (query().length === 0) { + search?.actions.getRecomendedUsers(); + return; + } + + search?.actions.findUsers(query()); + }); + + createEffect(() => { + if (isMentioning()) { + + mentionPositionOptions(); + + if (search?.users && search.users.length > 0) { + setHighlightedUser(0); + } + } + }); + + + const mentionPositionOptions = () => { + if (!newMessageInput || !mentionOptions || !newMessageWrapper) { + return; + } + + const taRect = newMessageInput.getBoundingClientRect(); + + let newBottom = taRect.height - mentionCursorPosition.top; + let newLeft = mentionCursorPosition.left; + + mentionOptions.style.bottom = `${newBottom}px`; + mentionOptions.style.left = `${newLeft}px`; + }; + + const selectEmoji = (emoji: EmojiOption) => { + if (!newMessageInput) { + return; + } + + const msg = message(); + + // Get cursor position to determine insertion point + let cursor = newMessageInput.selectionStart; + + // Get index of the token and insert emoji character + const index = msg.slice(0, cursor).lastIndexOf(':'); + const value = msg.slice(0, index) + emoji.char + msg.slice(cursor); + + // Reset query, update message and text area value + setMessage(value); + newMessageInput.value = message(); + + // Calculate new cursor position + newMessageInput.selectionEnd = index + 1; + newMessageInput.focus(); + + setEmojiInput(false); + setEmojiQuery(''); + setEmojiResults(() => []); + + // Dispatch input event to recalculate UI position + // const e = new Event('input', { bubbles: true, cancelable: true}); + // newMessageInput.dispatchEvent(e); + }; + + + const [userRefs, setUserRefs] = createStore>({}); + + + const selectUser = (user: PrimalUser | undefined) => { + if (!newMessageInput || !user) { + return; + } + + setMentioning(false); + + const name = userName(user); + + setUserRefs((refs) => ({ + ...refs, + [name]: user, + })); + + const msg = message(); + + // Get cursor position to determine insertion point + let cursor = newMessageInput.selectionStart; + + // Get index of the token and inster user's handle + const index = msg.slice(0, cursor).lastIndexOf('@'); + const value = msg.slice(0, index) + `@\`${name}\`` + msg.slice(cursor); + + // Reset query, update message and text area value + setQuery(''); + setMessage(value); + newMessageInput.value = message(); + + newMessageInput.focus(); + + // Calculate new cursor position + cursor = value.slice(0, cursor).lastIndexOf('@') + name.length + 3; + newMessageInput.selectionEnd = cursor; + + + // Dispatch input event to recalculate UI position + const e = new Event('input', { bubbles: true, cancelable: true}); + newMessageInput.dispatchEvent(e); + }; + // const selectUser = (user: PrimalUser) => { + + // if (!newMessageInput) { + // return; + // } + // const name = userName(user); + + // setUserRefs((refs) => ({ + // ...refs, + // [name]: user, + // })); + + // messages?.actions.addUserReference(user); + + // let value = message(); + + // value = value.slice(0, value.lastIndexOf('@')); + + // setQuery(''); + + // setMessage(`${value}@\`${name}\` `); + // newMessageInput.value = message(); + + // newMessageInput.focus(); + + + // // Dispatch input event to recalculate UI position + // const e = new Event('input', { bubbles: true, cancelable: true}); + // newMessageInput.dispatchEvent(e); + // }; + + createEffect(() => { + if (account?.hasPublicKey()) { + profile?.actions.setProfileKey(account.publicKey) + } + }); + + createEffect(() => { + if (messages?.selectedSender) { + + const element = document.querySelector(`[data-user="${messages.selectedSender.pubkey}"]`); + + if (element && sendersListElement && !isVisibleInContainer(element, sendersListElement)) { + element.scrollIntoView(); + } + + } + }); + + createEffect(() => { + if (emojiQuery().length > emojiSearchLimit) { + setEmojiResults(() => emojiSearch(emojiQuery())); + } + }); + + createEffect(() => { + if (isEmojiInput()) { + emojiPositionOptions(); + + if (emojiResults.length > 0) { + setHighlightedEmoji(0); + } + } + }); + + const emojiPositionOptions = () => { + if (!newMessageInput || !emojiOptions || !newMessageWrapper) { + return; + } + + const taRect = newMessageInput.getBoundingClientRect(); + + let newBottom = taRect.height - emojiCursorPosition.top; + let newLeft = emojiCursorPosition.left; + + emojiOptions.style.bottom = `${newBottom}px`; + emojiOptions.style.left = `${newLeft}px`; + }; + + + const onInput = () => { + newMessageInput && setMessage(newMessageInput.value) + } + + return ( +
+ + + + + + {}} + noLinks={true} + hideDefault={true} + onUserSelect={addUserToSenders} + /> + + +
+
+ {intl.formatMessage(tMessages.title)} +
+
+ +
+ +
+
+ +
+ +
+ +
+ +
+ + { + (sender) => ( + + ) + } + +
+ +
+
+
+ +
+ + + +
+ + {(user, index) => ( + } + statNumber={search?.scores[user.pubkey]} + statLabel={intl.formatMessage(tSearch.followers)} + onClick={() => selectUser(user)} + highlighted={highlightedUser() === index()} + /> + )} + +
+
+ + emojiSearchLimit}> +
+ + {(emoji, index) => ( + + )} + +
+
+
+
+ + + {messages?.isConversationLoaded ? + <> : + + } + } + > + {(thread) => ( + + + + +
+ + {(msg) => ( +
+ )} +
+
+ +
+ {date(thread.messages[0].created_at, 'long', messages?.now).label} +
+
+
+ } + > +
+ + + +
+ + {(msg) => ( +
+ )} +
+
+ +
+ {date(thread.messages[0].created_at, 'long', messages?.now).label} +
+
+
+ + )} + + + + +
+
+
+ + ); +} + +export default Messages; diff --git a/src/pages/NotFound.module.scss b/src/pages/NotFound.module.scss new file mode 100644 index 0000000..47a93f3 --- /dev/null +++ b/src/pages/NotFound.module.scss @@ -0,0 +1,3 @@ +.message { + color: var(--text-secondary) +} diff --git a/src/pages/NotFound.tsx b/src/pages/NotFound.tsx new file mode 100644 index 0000000..63503d7 --- /dev/null +++ b/src/pages/NotFound.tsx @@ -0,0 +1,21 @@ +import { useIntl } from '@cookbook/solid-intl'; +import { Component } from 'solid-js'; +import MissingPage from '../components/MissingPage/MissingPage'; +import { placeholders } from '../translations'; + +import styles from './NotFound.module.scss'; + +const NotFound: Component = () => { + + const intl = useIntl(); + + return ( + +

+ {intl.formatMessage(placeholders.pageNotFound)} +

+
+ ); +} + +export default NotFound; diff --git a/src/pages/Notifications.module.scss b/src/pages/Notifications.module.scss new file mode 100644 index 0000000..f7142ac --- /dev/null +++ b/src/pages/Notifications.module.scss @@ -0,0 +1,95 @@ +.fullHeader { + display: grid; + height: 120px; + align-items: center; + justify-content: left; + + >div { + font-weight: 300; + font-size: 32px; + line-height: 34px; + color: var(--brand-text); + text-transform: lowercase; + } +} + +.separator { + width: 100%; + height: 1px; + background-color: var(--text-primary); +} + +.oldNotifications { + position: relative; +} + +.loader { + margin-top: 120px; + position: relative; +} + +.newContentNotification { + position: fixed; + top: 42px; + left: calc(calc(100vw - 1240px) / 2 + 176px + 32px); + width: 640px; + z-index: 20; + display: flex; + justify-content: center; + + >button { + display: flex; + align-items: center; + justify-content: center; + width: auto; + height: 40px; + background: var(--brand-gradient); + font-weight: 400; + font-size: 14px; + line-height: 18px; + border: none; + border-radius: 20px; + padding-block: 0; + padding-left: 2px; + margin: 0; + + .avatars { + display: flex; + align-items: center; + height: 40px; + .avatar { + border: solid 2px var(--text-primary); + border-radius: 50%; + width: 36px; + height: 36px; + transition: margin-right 0.2s; + margin-right: -16px; + } + } + + .counter { + margin-left: 24px; + } + } + +} + +@media only screen and (max-width: 1300px) { + .newContentNotification { + left: calc(calc(100vw - 1032px) / 2 + 48px + 32px); + } +} + +@media only screen and (max-width: 1087px) { + .newContentNotification { + left: calc(calc(100vw - 720px) / 2 + 48px + 32px); + } +} + +@media only screen and (max-width: 720px) { + .newContentNotification { + left: 0; + width: 100%; + justify-content: center; + } +} diff --git a/src/pages/Notifications.tsx b/src/pages/Notifications.tsx new file mode 100644 index 0000000..f7c6922 --- /dev/null +++ b/src/pages/Notifications.tsx @@ -0,0 +1,1154 @@ +import { useIntl } from '@cookbook/solid-intl'; +import { useSearchParams } from '@solidjs/router'; +import { nip19 } from 'nostr-tools'; +import { Component, createEffect, createMemo, createSignal, For, onCleanup, Show } from 'solid-js'; +import { createStore } from 'solid-js/store'; +import { APP_ID } from '../App'; +import Branding from '../components/Branding/Branding'; +import Loader from '../components/Loader/Loader'; +import NotificationItem from '../components/Notifications/NotificationItem'; +import NotificationItem2 from '../components/Notifications/NotificationItem2'; +import NotificationsSidebar from '../components/NotificatiosSidebar/NotificationsSidebar'; +import Paginator from '../components/Paginator/Paginator'; +import Search from '../components/Search/Search'; +import StickySidebar from '../components/StickySidebar/StickySidebar'; +import Wormhole from '../components/Wormhole/Wormhole'; +import { Kind, minKnownProfiles, NotificationType, notificationTypeUserProps } from '../constants'; +import { useAccountContext } from '../contexts/AccountContext'; +import { useNotificationsContext } from '../contexts/NotificationsContext'; +import { getLastSeen, getNotifications, getOldNotifications, setLastSeen, truncateNumber } from '../lib/notifications'; +import { subscribeTo } from '../sockets'; +import { convertToNotes } from '../stores/note'; +import { convertToUser, emptyUser } from '../stores/profile'; +import { FeedPage, NostrMentionContent, NostrNoteActionsContent, NostrNoteContent, NostrStatsContent, NostrUserContent, NostrUserStatsContent, NoteActions, PrimalNote, PrimalNotification, PrimalNotifUser, PrimalUser, SortedNotifications } from '../types/primal'; +import { notifications as t } from '../translations'; + +import styles from './Notifications.module.scss'; + +const Notifications: Component = () => { + + const account = useAccountContext(); + const notifications = useNotificationsContext(); + const intl = useIntl(); + + const [queryParams, setQueryParams] = useSearchParams(); + + const [notifSince, setNotifSince] = createSignal(); + + const [sortedNotifications, setSortedNotifications] = createStore({}); + + const [users, setUsers] = createStore>({}); + + const [userStats, setUserStats] = createStore>({}); + + const [allSet, setAllSet] = createSignal(false); + const [fetchingOldNotifs, setfetchingOldNotifs] = createSignal(false); + + + const newNotifCount = () => { + if (!notifications?.notificationCount) { + return 0; + } + + if (notifications.notificationCount > 100) { + return 100; + } + + return notifications.notificationCount; + }; + + type NotificationStore = { + notes: PrimalNote[], + users: PrimalUser[], + page: FeedPage, + reposts: Record | undefined, + } + + type OldNotificationStore = { + notes: PrimalNote[], + users: Record, + userStats: Record, + page: FeedPage & { notifications: PrimalNotification[]}, + reposts: Record | undefined, + notifications: PrimalNotification[], + } + + const [relatedNotes, setRelatedNotes] = createStore({ + notes: [], + users: [], + page: { messages: [], users: {}, postStats: {}, mentions: {}, noteActions: {} }, + reposts: {}, + }) + + const [oldNotifications, setOldNotifications] = createStore({ + notes: [], + users: {}, + userStats: {}, + page: { messages: [], users: {}, postStats: {}, notifications: [], mentions: {}, noteActions: {} }, + reposts: {}, + notifications: [], + }) + + const hasNewNotifications = createMemo(() => { + return Object.keys(sortedNotifications).length > 0; + }); + + const publicKey = () => { + const user = queryParams.user; + if (user) { + if (minKnownProfiles.names[user]) { + return minKnownProfiles.names[user]; + } + + if (user.startsWith('npub')) { + return nip19.decode(user).data; + } + + return user; + } + + return account?.publicKey; + } + + createEffect(() => { + const pk = publicKey(); + if (pk) { + const subid = `notif_ls_${APP_ID}` + + const unsub = subscribeTo(subid, async (type, _, content) => { + if (type === 'EVENT' && content?.kind === Kind.Timestamp) { + + const timestamp = parseInt(content.content); + + if (!isNaN(timestamp)) { + setNotifSince(timestamp); + } + + unsub(); + return; + } + + if (type === 'EOSE') { + if (!notifSince()) { + setNotifSince(0); + } + } + + }); + + getLastSeen(pk as string, subid); + } + }); + + createEffect(() => { + if (account?.hasPublicKey() && publicKey() === account.publicKey) { + const subid = `notif_sls_${APP_ID}`; + + const unsub = subscribeTo(subid, async (type, _, content) => { + if (type === 'EOSE') { + unsub(); + return; + } + + if (type === 'NOTICE') { + console.log('Error setting notifications lats seen'); + unsub(); + return; + } + + }); + + setTimeout(() => { + setLastSeen(subid, Math.floor((new Date()).getTime() / 1000)); + }, 1000); + + } + }); + + let newNotifs: Record = {}; + + // Fetch new notifications + const fetchNewNotifications = (pk: string) => { + const subid = `notif_${APP_ID}` + + const unsub = subscribeTo(subid, async (type, _, content) => { + if (type === 'EVENT') { + if (!content?.content) { + return; + } + + if (content.kind === Kind.Notification) { + + const notif = JSON.parse(content.content) as PrimalNotification; + + if (newNotifs[notif.type]) { + newNotifs[notif.type].push(notif); + } + else { + newNotifs[notif.type] = [notif]; + } + + return; + } + + if (content.kind === Kind.Metadata) { + const user = content as NostrUserContent; + + setUsers((usrs) => ({ ...usrs, [user.pubkey]: { ...user } })); + + setRelatedNotes('page', 'users', + (usrs) => ({ ...usrs, [user.pubkey]: { ...user } }) + ); + return; + } + + if (content.kind === Kind.UserStats) { + const stat = content as NostrUserStatsContent; + const statContent = JSON.parse(content.content); + + setUserStats((stats) => ({ ...stats, [stat.pubkey]: { ...statContent } })); + return; + } + + if ([Kind.Text, Kind.Repost].includes(content.kind)) { + const message = content as NostrNoteContent; + + setRelatedNotes('page', 'messages', + (msgs) => [ ...msgs, { ...message }] + ); + + return; + } + + if (content.kind === Kind.NoteStats) { + const statistic = content as NostrStatsContent; + const stat = JSON.parse(statistic.content); + + setRelatedNotes('page', 'postStats', + (stats) => ({ ...stats, [stat.event_id]: { ...stat } }) + ); + return; + } + + if (content.kind === Kind.Mentions) { + const mentionContent = content as NostrMentionContent; + const mention = JSON.parse(mentionContent.content); + + setRelatedNotes('page', 'mentions', + (mentions) => ({ ...mentions, [mention.id]: { ...mention } }) + ); + return; + } + + if (content.kind === Kind.NoteActions) { + const noteActionContent = content as NostrNoteActionsContent; + const noteActions = JSON.parse(noteActionContent.content) as NoteActions; + + setRelatedNotes('page', 'noteActions', + (actions) => ({ ...actions, [noteActions.event_id]: { ...noteActions } }) + ); + return; + } + + } + + if (type === 'EOSE') { + setSortedNotifications(() => newNotifs); + setRelatedNotes('notes', () => [...convertToNotes(relatedNotes.page)]) + setAllSet(true); + unsub(); + return; + } + + }); + + const since = queryParams.ignoreLastSeen ? 0 : notifSince(); + + newNotifs = {}; + getNotifications(account?.publicKey, pk as string, subid, since); + + }; + + createEffect(() => { + const pk = publicKey(); + + if (!pk || notifSince() === undefined) { + return; + } + + fetchNewNotifications(pk as string); + }); + + onCleanup(() => { + setLastNotification(undefined); + setOldNotifications('notifications', []); + setOldNotifications('page', () => ({ messages: [], users: {}, postStats: {}, notifications: [] })); + setNotifSince(0); + setSortedNotifications({}) + }); + + const sortNotifByRecency = (notifs: PrimalNotification[]) => { + return notifs.sort((a: PrimalNotification, b: PrimalNotification) => { + return b.created_at - a.created_at; + }); + } + + const fetchOldNotifications = (until: number) => { + const subid = `notif_old_${APP_ID}` + + const unsub = subscribeTo(subid, async (type, _, content) => { + if (type === 'EVENT') { + if (!content?.content) { + return; + } + + if (content.kind === Kind.Notification) { + const notif = JSON.parse(content.content) as PrimalNotification; + + const isLastNotif = + lastNotification()?.created_at === notif.created_at && + lastNotification()?.type === notif.type; + + if (!isLastNotif) { + setOldNotifications('page', 'notifications', + (notifs) => notifs ? [ ...notifs, notif] : [notif], + ); + } + + return; + } + + if (content.kind === Kind.Metadata) { + const user = content as NostrUserContent; + + setOldNotifications('page', 'users', (usrs) => ({ ...usrs, [user.pubkey]: { ...user } })); + return; + } + + if (content.kind === Kind.UserStats) { + const stat = content as NostrUserStatsContent; + const statContent = JSON.parse(content.content); + + setOldNotifications('userStats', (stats) => ({ ...stats, [stat.pubkey]: { ...statContent } })); + return; + } + + if ([Kind.Text, Kind.Repost].includes(content.kind)) { + const message = content as NostrNoteContent; + + setOldNotifications('page', 'messages', + (msgs) => [ ...msgs, { ...message }] + ); + + return; + } + + if (content.kind === Kind.NoteStats) { + const statistic = content as NostrStatsContent; + const stat = JSON.parse(statistic.content); + + setOldNotifications('page', 'postStats', + (stats) => ({ ...stats, [stat.event_id]: { ...stat } }) + ); + return; + } + + if (content.kind === Kind.Mentions) { + const mentionContent = content as NostrMentionContent; + const mention = JSON.parse(mentionContent.content); + + setOldNotifications('page', 'mentions', + (mentions) => ({ ...mentions, [mention.id]: { ...mention } }) + ); + return; + } + + if (content.kind === Kind.NoteActions) { + const noteActionContent = content as NostrNoteActionsContent; + const noteActions = JSON.parse(noteActionContent.content) as NoteActions; + + setOldNotifications('page', 'noteActions', + (actions) => ({ ...actions, [noteActions.event_id]: { ...noteActions } }) + ); + return; + } + + } + + if (type === 'EOSE') { + + // Sort notifications + const notifs = [...oldNotifications.page.notifications]; + + const sorted = sortNotifByRecency(notifs); + + setOldNotifications('notifications', (notifs) => [ ...notifs, ...sorted]) + + // Convert related notes + setOldNotifications('notes', (notes) => [...notes, ...convertToNotes(oldNotifications.page)]) + + const pageUsers = oldNotifications.page.users; + + const newUsers = Object.keys(pageUsers).reduce((acc, key) => { + return { ...acc, [pageUsers[key].pubkey]: { ...convertToUser(pageUsers[key])}}; + }, {}); + + setOldNotifications('users', (users) => ({ ...users, ...newUsers })); + + setfetchingOldNotifs(false); + unsub(); + return; + } + + }); + + setOldNotifications('page', () => ({ messages: [], users: {}, postStats: {}, notifications: [] })); + + const pk = publicKey(); + + if (pk) { + setfetchingOldNotifs(true); + getOldNotifications(account?.publicKey, pk as string, subid, until); + } + + } + + // Fetch old notifications + createEffect(() => { + if (account?.hasPublicKey() && !queryParams.ignoreLastSeen && notifSince() !== undefined) { + fetchOldNotifications(notifSince() || 0); + } + }); + + const getUsers = ( + notifs: PrimalNotification[], + type: NotificationType, + ) => { + const knownUsers = Object.keys(users); + const userProp = notificationTypeUserProps[type]; + + const pks = notifs.reduce((acc, n) => { + // @ts-ignore + const pubkey = n[userProp]; + + if (!pubkey) { + return acc; + } + return acc.includes(pubkey) ? acc : [...acc, pubkey]; + }, []); + + return pks.map((pk) => { + const user = knownUsers.includes(pk) ? + convertToUser(users[pk]) : + emptyUser(pk); + + return { ...user, ...userStats[pk]} as PrimalNotifUser; + }); + } + + const groupBy = (notifs: PrimalNotification[], keyName: string) => { + return notifs.reduce>( + (group: Record, notif) => { + // @ts-ignore + const key: string = notif[keyName] || 'none'; + + group[key] = group[key] ?? []; + group[key].push(notif); + + return group; + }, + {}, + ); + }; + + const newUserFollowedYou = () => { + const type = NotificationType.NEW_USER_FOLLOWED_YOU; + const notifs = sortedNotifications[type]; + + if (!notifs) { + return; + } + + return + }; + + const userUnfollowedYou = () => { + const type = NotificationType.USER_UNFOLLOWED_YOU; + const notifs = sortedNotifications[type]; + + if (!notifs) { + return; + } + + return + }; + + const yourPostWasLiked = () => { + const type = NotificationType.YOUR_POST_WAS_LIKED; + const notifs = sortedNotifications[type] || []; + + const grouped = groupBy(notifs, 'your_post'); + + const keys = Object.keys(grouped); + + return + {key => { + return ( + n.post.id === key)} + /> + )}} + + }; + + // + const yourPostWasReposted = () => { + const type = NotificationType.YOUR_POST_WAS_REPOSTED; + const notifs = sortedNotifications[type] || []; + + const grouped = groupBy(notifs, 'your_post'); + + const keys = Object.keys(grouped); + + return + {key => { + return ( + n.post.id === key)} + /> + )} + } + + }; + + const yourPostWasRepliedTo = () => { + const type = NotificationType.YOUR_POST_WAS_REPLIED_TO; + const notifs = sortedNotifications[type] || []; + + const grouped = groupBy(notifs, 'reply'); + + const keys = Object.keys(grouped); + + + return + {key => { + return ( + n.post.id === key)} + /> + )} + } + + }; + + const yourPostWasZapped = () => { + const type = NotificationType.YOUR_POST_WAS_ZAPPED; + const notifs = sortedNotifications[type] || []; + + const grouped = groupBy(notifs, 'your_post'); + + const keys = Object.keys(grouped); + + return + {key => { + const sats = grouped[key].reduce((acc, n) => { + return n.satszapped ? acc + n.satszapped : acc; + },0); + + return ( + n.post.id === key)} + iconInfo={`${truncateNumber(sats)}`} + iconTooltip={`${sats} sats`} + /> + )} + } + + }; + + const youWereMentioned = () => { + const type = NotificationType.YOU_WERE_MENTIONED_IN_POST; + const notifs = sortedNotifications[type] || []; + + const grouped = groupBy(notifs, 'you_were_mentioned_in'); + + const keys = Object.keys(grouped); + + const notes = relatedNotes.notes.filter(n => keys.includes(n.post.id)); + + if (notes.length === 0) { + return; + } + + const knownUsers = Object.keys(users); + + const rUsers: Record = notes.reduce((acc, note) => { + const pk = note.user.pubkey; + + const rUser = knownUsers.includes(pk) ? + convertToUser(users[pk]) : + emptyUser(pk); + + const usrs = [{...rUser, ...userStats[pk]}]; + + return { ...acc, [note.post.id]: usrs}; + + }, {}); + + + return + {key => { + return ( + n.post.id === key)} + /> + )} + } + + }; + + const yourPostWasMentioned = () => { + const type = NotificationType.YOUR_POST_WAS_MENTIONED_IN_POST; + const notifs = sortedNotifications[type] || []; + + const grouped = groupBy(notifs, 'your_post_were_mentioned_in'); + + const keys = Object.keys(grouped); + + + const notes = relatedNotes.notes.filter(n => keys.includes(n.post.id)); + + if (notes.length === 0) { + return; + } + + const knownUsers = Object.keys(users); + + const rUsers: Record = notes.reduce((acc, note) => { + const pk = note.user.pubkey; + + const rUser = knownUsers.includes(pk) ? + convertToUser(users[pk]) : + emptyUser(pk); + + const usrs = [{...rUser, ...userStats[pk]}]; + + return { ...acc, [note.post.id]: usrs}; + + }, {}); + + + return + {key => { + return ( + n.post.id === key)} + /> + )} + } + + }; + + const postYouWereMentionedInWasLiked = () => { + const type = NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_LIKED; + const notifs = sortedNotifications[type] || []; + + const grouped = groupBy(notifs, 'post_you_were_mentioned_in'); + + const keys = Object.keys(grouped); + + + const notes = relatedNotes.notes.filter(n => keys.includes(n.post.id)); + + if (notes.length === 0) { + return; + } + + const knownUsers = Object.keys(users); + + const rUsers: Record = notes.reduce((acc, note) => { + const pk = note.user.pubkey; + + const rUser = knownUsers.includes(pk) ? + convertToUser(users[pk]) : + emptyUser(pk); + + const usrs = [{...rUser, ...userStats[pk]}]; + + return { ...acc, [note.post.id]: usrs}; + + }, {}); + + + return + {key => { + return ( + n.post.id === key)} + /> + )} + } + + }; + + const postYouWereMentionedInWasZapped = () => { + const type = NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_ZAPPED; + const notifs = sortedNotifications[type] || []; + + const grouped = groupBy(notifs, 'post_you_were_mentioned_in'); + + const keys = Object.keys(grouped); + + + const notes = relatedNotes.notes.filter(n => keys.includes(n.post.id)); + + if (notes.length === 0) { + return; + } + + const knownUsers = Object.keys(users); + + const rUsers: Record = notes.reduce((acc, note) => { + const pk = note.user.pubkey; + + const rUser = knownUsers.includes(pk) ? + convertToUser(users[pk]) : + emptyUser(pk); + + const usrs = [{...rUser, ...userStats[pk]}]; + + return { ...acc, [note.post.id]: usrs}; + + }, {}); + + + return + {key => { + const sats = grouped[key].reduce((acc, n) => { + return n.satszapped ? acc + n.satszapped : acc; + },0); + return ( + n.post.id === key)} + iconInfo={`${truncateNumber(sats)}`} + iconTooltip={`${sats} sats`} + /> + )} + } + + }; + + const postYouWereMentionedInWasReposted = () => { + const type = NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_REPOSTED; + const notifs = sortedNotifications[type] || []; + + const grouped = groupBy(notifs, 'post_you_were_mentioned_in'); + + const keys = Object.keys(grouped); + + const notes = relatedNotes.notes.filter(n => keys.includes(n.post.id)); + + if (notes.length === 0) { + return; + } + + const knownUsers = Object.keys(users); + + const rUsers: Record = notes.reduce((acc, note) => { + const pk = note.user.pubkey; + + const rUser = knownUsers.includes(pk) ? + convertToUser(users[pk]) : + emptyUser(pk); + + const usrs = [{...rUser, ...userStats[pk]}]; + + return { ...acc, [note.post.id]: usrs}; + + }, {}); + + + return + {key => { + return ( + n.post.id === key)} + /> + )} + } + + }; + + const postYouWereMentionedInWasRepliedTo = () => { + const type = NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_REPLIED_TO; + const notifs = sortedNotifications[type] || []; + + const grouped = groupBy(notifs, 'reply'); + + const keys = Object.keys(grouped); + + const notes = relatedNotes.notes.filter(n => keys.includes(n.post.id)); + + if (notes.length === 0) { + return; + } + + const knownUsers = Object.keys(users); + + const rUsers: Record = notes.reduce((acc, note) => { + const pk = note.user.pubkey; + + const rUser = knownUsers.includes(pk) ? + convertToUser(users[pk]) : + emptyUser(pk); + + const usrs = [{...rUser, ...userStats[pk]}]; + + return { ...acc, [note.post.id]: usrs}; + + }, {}); + + + return + {key => { + return ( + n.post.id === key)} + /> + )} + } + + }; + + + const postYourPostWasMentionedInWasLiked = () => { + const type = NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_LIKED; + const notifs = sortedNotifications[type] || []; + + const grouped = groupBy(notifs, 'post_your_post_was_mentioned_in'); + + const keys = Object.keys(grouped); + + const notes = relatedNotes.notes.filter(n => keys.includes(n.post.id)); + + if (notes.length === 0) { + return; + } + + const knownUsers = Object.keys(users); + + const rUsers: Record = notes.reduce((acc, note) => { + const pk = note.user.pubkey; + + const rUser = knownUsers.includes(pk) ? + convertToUser(users[pk]) : + emptyUser(pk); + + const usrs = [{...rUser, ...userStats[pk]}]; + + return { ...acc, [note.post.id]: usrs}; + + }, {}); + + + return + {key => { + return ( + n.post.id === key)} + /> + )} + } + + }; + + const postYourPostWasMentionedInWasZapped = () => { + const type = NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_ZAPPED; + const notifs = sortedNotifications[type] || []; + + const grouped = groupBy(notifs, 'post_your_post_was_mentioned_in'); + + const keys = Object.keys(grouped); + + const notes = relatedNotes.notes.filter(n => keys.includes(n.post.id)); + + if (notes.length === 0) { + return; + } + + const knownUsers = Object.keys(users); + + const rUsers: Record = notes.reduce((acc, note) => { + const pk = note.user.pubkey; + + const rUser = knownUsers.includes(pk) ? + convertToUser(users[pk]) : + emptyUser(pk); + + const usrs = [{...rUser, ...userStats[pk]}]; + + return { ...acc, [note.post.id]: usrs}; + + }, {}); + + + return + {key => { + const sats = grouped[key].reduce((acc, n) => { + return n.satszapped ? acc + n.satszapped : acc; + },0); + return ( + n.post.id === key)} + iconInfo={`${truncateNumber(sats)}`} + iconTooltip={`${sats} sats`} + /> + )} + } + + }; + + const postYourPostWasMentionedInWasReposted = () => { + const type = NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPOSTED; + const notifs = sortedNotifications[type] || []; + + const grouped = groupBy(notifs, 'post_your_post_was_mentioned_in'); + + const keys = Object.keys(grouped); + + const notes = relatedNotes.notes.filter(n => keys.includes(n.post.id)); + + if (notes.length === 0) { + return; + } + + const knownUsers = Object.keys(users); + + const rUsers: Record = notes.reduce((acc, note) => { + const pk = note.user.pubkey; + + const rUser = knownUsers.includes(pk) ? + convertToUser(users[pk]) : + emptyUser(pk); + + const usrs = [{...rUser, ...userStats[pk]}]; + + return { ...acc, [note.post.id]: usrs}; + + }, {}); + + + return + {key => { + return ( + n.post.id === key)} + /> + )} + } + + }; + + const postYourPostWasMentionedInWasRepliedTo = () => { + const type = NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPLIED_TO; + const notifs = sortedNotifications[type] || []; + + const grouped = groupBy(notifs, 'reply'); + + const keys = Object.keys(grouped); + + const notes = relatedNotes.notes.filter(n => keys.includes(n.post.id)); + + if (notes.length === 0) { + return; + } + + const knownUsers = Object.keys(users); + + const rUsers: Record = notes.reduce((acc, note) => { + const pk = note.user.pubkey; + + const rUser = knownUsers.includes(pk) ? + convertToUser(users[pk]) : + emptyUser(pk); + + const usrs = [{...rUser, ...userStats[pk]}]; + + return { ...acc, [note.post.id]: usrs}; + + }, {}); + + + return + {key => { + return ( + n.post.id === key)} + /> + )} + } + + }; + + const [lastNotification, setLastNotification] = createSignal(); + + const fetchMoreNotifications = () => { + const lastNotif = oldNotifications.notifications[oldNotifications.notifications.length - 1]; + + if (!lastNotif || lastNotif.created_at === lastNotification()?.created_at) { + return; + } + + setLastNotification(lastNotif); + + const until = lastNotif.created_at; + + if (until > 0) { + fetchOldNotifications(until); + } + } + + const loadNewContent = () => { + fetchNewNotifications(publicKey() as string); + setLastSeen(`notif_sls_${APP_ID}`, Math.floor((new Date()).getTime() / 1000)); + } + + return ( +
+ + + + + + + + +
+
+ {intl.formatMessage(t.title)} +
+
+ + + 0 && !account?.showNewNoteForm}> +
+ +
+
+ + + + +
+ } + > + + + + + {newUserFollowedYou()} + {userUnfollowedYou()} + + {yourPostWasZapped()} + + {yourPostWasRepliedTo()} + {yourPostWasReposted()} + {yourPostWasLiked()} + + {youWereMentioned()} + {yourPostWasMentioned()} + + {postYouWereMentionedInWasZapped()} + {postYouWereMentionedInWasRepliedTo()} + {postYouWereMentionedInWasReposted()} + {postYouWereMentionedInWasLiked()} + + {postYourPostWasMentionedInWasZapped()} + {postYourPostWasMentionedInWasRepliedTo()} + {postYourPostWasMentionedInWasReposted()} + {postYourPostWasMentionedInWasLiked()} + + +
+
+ + +
+ +
+
+ + 0}> +
+ + {notif => ( + + )} + + +
+
+ + + + ); +} + +export default Notifications; diff --git a/src/pages/Profile.module.scss b/src/pages/Profile.module.scss new file mode 100644 index 0000000..2b5c8ed --- /dev/null +++ b/src/pages/Profile.module.scss @@ -0,0 +1,418 @@ +.fullHeader { + position: relative; + background-color: var(--background-card); + padding-bottom: 20px; + border-radius: 0 0 8px 8px; +} + +.banner { + width: 100%; + height: 214px; + + >img { + width: 100%; + height: 214px; + object-fit: cover; + } +} + +.userImage { + position: absolute; + top: 148px; + left: 15px; + .avatar { + border: solid 4px var(--background-card); + border-radius: 50%; + background-color: var(--background-card); + } +} + +.verifiedIconL { + width: 22px; + height: 22px; + display: inline-block; + margin: 0px 12px 0px 6px; + background-color: var(--accent-2); + -webkit-mask: url(../assets/icons/verified.svg) no-repeat 0px / 22px; + mask: url(../assets/icons/verified.svg) no-repeat 0px / 22px; +} + +.verifiedIconS { + width: 12px; + height: 12px; + display: inline-block; + margin: 0px 2px; + background-color: var(--text-tertiary-2); + -webkit-mask: url(../assets/icons/verified.svg) no-repeat 0px / 12px; + mask: url(../assets/icons/verified.svg) no-repeat 0px / 12px; +} + +.keyIcon { + min-width: 16px; + height: 16px; + display: inline-block; + margin: 0px; + background-color: var(--text-tertiary-2); + -webkit-mask: url(../assets/icons/key.svg) no-repeat 0 0 / 16px 16px; + mask: url(../assets/icons/key.svg) no-repeat 0 0 / 16px 16px; +} + +.copyIcon { + width: 16px; + height: 16px; + display: inline-block; + margin: 0px 2px; + background-color: var(--text-tertiary-2); + -webkit-mask: url(../assets/icons/copy.svg) no-repeat 0 0 / 16px 16px; + mask: url(../assets/icons/copy.svg) no-repeat 0 0 / 16px 16px; +} + +.linkIcon { + width: 16px; + height: 16px; + display: inline-block; + margin: 0px 6px 0px 0px; + background-color: var(--text-tertiary-2); + -webkit-mask: url(../assets/icons/link.svg) no-repeat 0 0 / 16px 16px; + mask: url(../assets/icons/link.svg) no-repeat 0 0 / 16px 16px; +} + +.profileActions { + display: flex; + justify-content: right; + padding-top: 20px; + padding-right: 8px; + padding-bottom: 36px; +} + +@mixin smallButton { + display: flex; + align-items: center; + justify-content: center; + width: 40px; + height: 40px; + border: none; + border-radius: 6px; + margin: 0px 6px; + padding: 0px; +} + +.smallSecondaryButton { + @include smallButton(); + + background-color: var(--background-card); + background: linear-gradient(var(--background-card), var(--background-card)) padding-box, + var(--brand-gradient-vertical) border-box; + border: 1px solid transparent; +} + +.smallPrimaryButton { + @include smallButton(); + background: var(--brand-gradient-vertical); + + .addFeedIcon { + background-color: white; + } +} + +.zapIcon { + width: 18px; + height: 18px; + display: inline-block; + margin: 0px; + background-color: var(--text-secondary); + -webkit-mask: url(../assets/icons/zaps.svg) no-repeat 0px / 18px; + mask: url(../assets/icons/zaps.svg) no-repeat 0px / 18px; +} + +.messageIcon { + width: 18px; + height: 18px; + display: inline-block; + margin: 0px; + background-color: var(--text-secondary); + -webkit-mask: url(../assets/icons/messages.svg) no-repeat 0px / 18px; + mask: url(../assets/icons/messages.svg) no-repeat 0px / 18px; +} + +.addFeedIcon { + width: 18px; + height: 18px; + display: inline-block; + margin: 0px; + background-color: var(--text-secondary); + -webkit-mask: url(../assets/icons/feed_add.svg) no-repeat 0px / 18px; + mask: url(../assets/icons/feed_add.svg) no-repeat 0px / 18px; +} + +.removeFeedIcon { + width: 18px; + height: 18px; + display: inline-block; + margin: 0px; + background-color: var(--text-secondary); + -webkit-mask: url(../assets/icons/feed_remove.svg) no-repeat 0px / 18px; + mask: url(../assets/icons/feed_remove.svg) no-repeat 0px / 18px; +} + +.primaryButton { + width: 90px; + height: 40px; + border: none; + border-radius: 6px; + margin: 0px 8px; + padding: 0px; + font-size: 16px; + line-height: 20px; + font-weight: 700; + background: var(--brand-gradient-vertical); + color: white; +} + + +.profileVerification { + margin-top: 12px; + margin-inline: 20px; + .avatarName { + display: flex; + align-items: center; + color: var(--text-primary); + font-size: 20px; + line-height: 20px; + font-weight: 700; + height: 26px; + } + .verificationInfo { + display: flex; + align-items: center; + color: var(--text-tertiary-2); + font-weight: 400; + font-size: 12px; + line-height: 16px; + margin-top: 10px; + .verified { + display: flex; + align-items: center; + .nip05 { + margin-right: 36px; + } + } + } + .publicKey { + display: flex; + height: 16px; + align-items: center; + color: var(--text-tertiary-2); + font-weight: 400; + font-size: 12px; + line-height: 16px; + + .npub { + color: var(--text-tertiary-2); + text-align: left; + border: none; + margin: 0; + padding: 0; + background-color: unset; + text-overflow: ellipsis; + white-space: nowrap; + overflow: hidden; + font-weight: 400; + font-size: 12px; + line-height: 16px; + display: flex; + + .copyIcon { + background-color: var(--accent-2); + } + + &:hover { + .copyIcon { + background-color: var(--text-primary); + } + } + } + } + +} + +.profileLinks { + display: flex; + align-items: center; + justify-content: space-between; + margin-inline: 20px; + margin-block: 12px; + + .website { + font-weight: 400; + font-size: 14px; + line-height: 20px; + display: flex; + align-items: center; + } + + .joined { + font-weight: 400; + font-size: 14px; + line-height: 20px; + text-align: right; + color: var(--text-tertiary-2); + } +} + +.profileAbout { + margin-top: 12px; + margin-inline: 20px; + font-weight: 400; + font-size: 14px; + line-height: 20px; + color: var(--text-primary); +} + +.userStats { + display: flex; + justify-content: left; + align-items: center; + margin-inline: 20px; + + .userStat { + display: flex; + justify-content: left; + align-items: baseline; + margin-right: 28px; + + .statNumber { + font-weight: 400; + font-size: 34px; + line-height: 34px; + color: var(--text-primary); + } + + .statName { + font-weight: 300; + font-size: 16px; + line-height: 19px; + color: var(--text-tertiary-2); + margin-left: 8px; + text-transform: lowercase; + } + + } +} + +.userFeed { + position: relative; +} + +.bannerPlaceholder { + width: 100%; + height: 214px; + background-color: var(--background-input); +} + +.followsBadge { + background-color: var(--background-input); + width: 80px; + height: 16px; + border-radius: 4px; + font-weight: 300; + font-size: 12px; + line-height: 16px; + color: var(--text-tertiary-2); + text-transform: lowercase; + display: flex; + align-items: center; + justify-content: center; +} + +.phoneAvatar { + display: none; +} +.desktopAvatar { + display: block; +} +.addToFeedButton { + display: block; +} + +@media only screen and (max-width: 720px) { + + .banner { + width: 100%; + height: 125px; + + >img { + width: 100%; + height: 125px; + object-fit: cover; + } + } + + .userImage { + position: absolute; + top: 96px; + left: 15px; + } + + .phoneAvatar { + display: block; + } + .desktopAvatar { + display: none; + } + .addToFeedButton { + display: none; + } + .joined { + display: none; + } + + .profileActions { + padding-top: 12px; + padding-bottom: 0px; + } + + .userStats { + justify-content: space-between; + .userStat { + margin-right: 15px; + flex-direction: column; + justify-content: center; + align-items: center; + + .statNumber { + font-size: 28px; + line-height: 34px; + } + + .statName { + font-size: 14px; + line-height: 16px; + } + + } + } + + .profileVerification { + .verificationInfo { + flex-direction: column; + justify-content: flex-start; + align-items: start; + >div { + margin-bottom: 12px; + font-weight: 400; + font-size: 14px; + line-height: 16px; + } + } + } + .profileAbout { + margin-top: 0px; + } +} + +.cacheFlag { + img { + border: 1px solid red; + } +} diff --git a/src/pages/Profile.tsx b/src/pages/Profile.tsx new file mode 100644 index 0000000..6128402 --- /dev/null +++ b/src/pages/Profile.tsx @@ -0,0 +1,404 @@ +import { RouteDataFuncArgs, useNavigate, useParams, useRouteData } from '@solidjs/router'; +import { nip19 } from 'nostr-tools'; +import { + Component, + createEffect, + createMemo, + createSignal, + For, + Resource, + Show +} from 'solid-js'; +import Avatar from '../components/Avatar/Avatar'; +import Branding from '../components/Branding/Branding'; +import Note from '../components/Note/Note'; +import { hexToNpub } from '../lib/keys'; +import { humanizeNumber } from '../lib/stats'; +import { nip05Verification, truncateNpub } from '../stores/profile'; +import Paginator from '../components/Paginator/Paginator'; +import { useToastContext } from '../components/Toaster/Toaster'; +import { useSettingsContext } from '../contexts/SettingsContext'; +import { useProfileContext } from '../contexts/ProfileContext'; +import { useAccountContext } from '../contexts/AccountContext'; +import Wormhole from '../components/Wormhole/Wormhole'; +import { useIntl } from '@cookbook/solid-intl'; +import { urlify, sanitize, replaceLinkPreviews } from '../lib/notes'; +import { shortDate } from '../lib/dates'; + +import styles from './Profile.module.scss'; +import StickySidebar from '../components/StickySidebar/StickySidebar'; +import ProfileSidebar from '../components/ProfileSidebar/ProfileSidebar'; +import { VanityProfiles } from '../types/primal'; +import PageTitle from '../components/PageTitle/PageTitle'; +import FollowButton from '../components/FollowButton/FollowButton'; +import Search from '../components/Search/Search'; +import { useMediaContext } from '../contexts/MediaContext'; +import { profile as t, actions as tActions } from '../translations'; + +const Profile: Component = () => { + + const settings = useSettingsContext(); + const toaster = useToastContext(); + const profile = useProfileContext(); + const account = useAccountContext(); + const media = useMediaContext(); + const intl = useIntl(); + const navigate = useNavigate(); + + const params = useParams(); + + const routeData = useRouteData<(opts: RouteDataFuncArgs) => Resource>(); + + const getHex = () => { + if (params.vanityName && routeData()) { + const name = params.vanityName.toLowerCase(); + const hex = routeData()?.names[name]; + + if (hex) { + return hex; + } + + navigate('/404'); + } + + if (params.vanityName) { + return ''; + } + + let hex = params.npub || account?.publicKey; + + if (params.npub?.startsWith('npub')) { + hex = nip19.decode(params.npub).data as string; + } + + return hex; + } + + const setProfile = (hex: string | undefined) => { + // if (hex === profile?.profileKey) { + // return; + // } + + profile?.actions.setProfileKey(hex); + profile?.actions.clearNotes(); + profile?.actions.fetchNotes(hex); + } + + // const react = createReaction(() => { + // setProfile(getHex()); + // }); + + // onMount(() => { + // // If connection doesn't exist at mount time, + // // create a one-time reaction, when connection is established + // // to fetch profile data. + // if (!isConnected()) { + // react(() => isConnected()); + // return; + // } + + // // Otherwise, fetch profile data. + // setProfile(getHex()); + // }); + + createEffect(() => { + if (account?.isKeyLookupDone) { + setProfile(getHex()); + } + }); + + const profileNpub = createMemo(() => { + return hexToNpub(profile?.profileKey); + }); + + const profileName = () => { + return profile?.userProfile?.displayName || + profile?.userProfile?.name || + truncateNpub(profileNpub()); + } + + const addToHome = () => { + const feed = { + name: `${profileName()}'s feed`, + hex: profile?.profileKey, + npub: profileNpub(), + }; + + settings?.actions.addAvailableFeed(feed); + toaster?.sendSuccess(`${profileName()}'s feed added to home page`); + }; + + const removeFromHome = () => { + const feed = { + name: `${profileName()}'s feed`, + hex: profile?.profileKey, + npub: profileNpub(), + }; + + settings?.actions.removeAvailableFeed(feed); + toaster?.sendSuccess(`${profileName()}'s feed removed from home page`); + }; + + const hasFeedAtHome = () => { + return !!settings?.availableFeeds.find(f => f.hex === profile?.profileKey); + }; + + const copyNpub = () => { + navigator.clipboard.writeText(profile?.userProfile?.npub || profileNpub()); + } + + const imgError = (event: any) => { + // Temprary solution until we decide what to to when banner is missing. + + // const image = event.target; + // image.onerror = ""; + // image.src = defaultAvatar; + + const banner = document.getElementById('profile_banner'); + + if (banner) { + banner.innerHTML = `
`; + } + + return true; + } + + const rectifyUrl = (url: string) => { + if (!url.startsWith('http://') && !url.startsWith('https://')) { + return `http://${url}`; + } + + return url; + } + + const onNotImplemented = () => { + toaster?.notImplemented(); + } + + const isFollowingYou = () => { + return account?.publicKey && profile?.following.includes(account.publicKey); + } + + const [isBannerCached, setisBannerCached] = createSignal(false); + + const banner = () => { + const src= profile?.userProfile?.banner; + const url = media?.actions.getMediaUrl(src, 'm', true); + + setisBannerCached(!!url); + + return url ?? src; + } + + const flagBannerForWarning = () => { + const dev = JSON.parse(localStorage.getItem('devMode') || 'false'); + + // @ts-ignore + if (isBannerCached() || !dev) { + return ''; + } + + return styles.cacheFlag; + } + + return ( + <> + + + + + + + + + + + + + + +
+
+
} + > + + +
+ + +
+
+
+ +
+ +
+ +
+
+
+
+ +
+ + + +
+ +
+ + } + > + +
+
+ + + +
+ + +
+
+ {profileName()} + +
+
+ +
+ {intl.formatMessage(t.followsYou)} +
+
+
+
+ +
+
+
{nip05Verification(profile?.userProfile)}
+
+
+
+
+ +
+
+
+
+ +
+
+ +
+ +
+ + {intl.formatMessage( + t.jointDate, + { + date: shortDate(profile?.userStats.time_joined), + }, + )} + +
+
+ +
+
+
+ {humanizeNumber(profile?.userStats?.follows_count || 0)} +
+
+ {intl.formatMessage(t.stats.follow)} +
+
+
+
+ {humanizeNumber(profile?.userStats?.followers_count || 0)} +
+
+ {intl.formatMessage(t.stats.followers)} +
+
+
+
+ {humanizeNumber(profile?.userStats?.note_count || 0)} +
+
+ {intl.formatMessage(t.stats.notes)} +
+
+ +
+ + + +
+ + {note => ( + + )} + + +
+ + ) +} + +export default Profile; diff --git a/src/pages/Search.module.scss b/src/pages/Search.module.scss new file mode 100644 index 0000000..99b312c --- /dev/null +++ b/src/pages/Search.module.scss @@ -0,0 +1,78 @@ +.fullHeader { + display: grid; + height: 120px; + align-items: center; + justify-content: left; + margin-bottom: -3px; + position: relative; + + .caption { + font-weight: 300; + font-size: 32px; + line-height: 34px; + color: var(--brand-text); + text-transform: lowercase; + } + + .addToFeed { + display: flex; + position: absolute; + bottom: 0px; + width: 100%; + height: 35px; + justify-content: flex-end; + align-items: flex-end; + + .noAdd { + display: flex; + align-items: center; + font-size: 16px; + line-height: 25px; + font-weight: 400; + color: var(--text-primary); + opacity: 0.6; + transition: opacity 0.4s; + } + + .addButton { + display: flex; + align-items: center; + margin: 0; + padding: 0; + border: none; + background-color: unset; + width: auto; + font-size: 16px; + line-height: 25px; + font-weight: 400; + color: var(--text-primary); + opacity: 0.6; + transition: opacity 0.4s; + + >span { + font-weight: 800; + margin-right: 5px; + } + + &:hover { + opacity: 1; + transition: opacity 0.4s; + } + + &:focus { + box-shadow: none; + } + } + } +} +.searchContent { + position: relative; +} + +.noResults { + font-weight: 300; + font-size: 22px; + line-height: 24px; + color: var(--text-tertiary); + text-transform: lowercase; +} diff --git a/src/pages/Search.tsx b/src/pages/Search.tsx new file mode 100644 index 0000000..6addbc3 --- /dev/null +++ b/src/pages/Search.tsx @@ -0,0 +1,136 @@ +import { + Component, + createEffect, + For, + Show, +} from 'solid-js'; +import Note from '../components/Note/Note'; +import Branding from '../components/Branding/Branding'; +import Wormhole from '../components/Wormhole/Wormhole'; +import StickySidebar from '../components/StickySidebar/StickySidebar'; +import { useAccountContext } from '../contexts/AccountContext'; +import { useIntl } from '@cookbook/solid-intl'; +import { isConnected } from '../sockets'; +import { useParams } from '@solidjs/router'; +import styles from './Search.module.scss'; +import { useSearchContext } from '../contexts/SearchContext'; +import SearchSidebar from '../components/SearchSidebar/SearchSidebar'; +import Loader from '../components/Loader/Loader'; +import { useToastContext } from '../components/Toaster/Toaster'; +import { useSettingsContext } from '../contexts/SettingsContext'; +import SearchComponent from '../components/Search/Search'; +import { toast as t, search as tSearch, actions as tActions } from '../translations'; + +const Search: Component = () => { + const params = useParams(); + const search = useSearchContext(); + const account = useAccountContext(); + const toaster = useToastContext(); + const settings = useSettingsContext(); + const intl = useIntl(); + + const query = () => decodeURI(params.query).replaceAll('%23', '#'); + + createEffect(() => { + if (isConnected() && query().length > 0 && search?.contentQuery !== query()) { + search?.actions.setContentQuery(query()); + search?.actions.findContent(query()); + search?.actions.findContentUsers(query(), account?.publicKey); + } + }); + + const hasFeedAtHome = () => { + const hex = `search;${decodeURI(params.query)}`; + + return !!settings?.availableFeeds.find(f => f.hex === hex); + }; + + const addToHomeFeed = () => { + const q = decodeURI(params.query).replaceAll('%23', '#') + const hex = `search;${q}`; + const name = intl.formatMessage( + tSearch.feedLabel, + { query: q || '' }, + ); + + const feed = { name, hex }; + + settings?.actions.addAvailableFeed(feed); + + toaster?.sendSuccess(intl.formatMessage( + t.addFeedToHomeSuccess, + { name }, + )); + }; + + return ( + <> + + + + + + + + + + + + +
+
+ {intl.formatMessage( + tSearch.title, + { query: query() || '' }, + )} +
+
+ + {intl.formatMessage(tActions.disabledAddFeedToHome)} +
+ } + > + + +
+ + +
+ } + > + 0} + fallback={ +
+ { + intl.formatMessage(tSearch.noResults) + } +
+ } + > + + {note => } + +
+
+
+ + ) +} + +export default Search; diff --git a/src/pages/Settings.module.scss b/src/pages/Settings.module.scss new file mode 100644 index 0000000..567efbc --- /dev/null +++ b/src/pages/Settings.module.scss @@ -0,0 +1,42 @@ +.settingsContainer { + background-color: var(--background-card); + padding-inline: 28px; + min-height: 100vh; + padding-bottom: 20px; +} + +.fullHeader { + display: grid; + height: 128px; + align-items: center; + justify-content: left; + + >div { + font-weight: 300; + font-size: 32px; + line-height: 34px; + color: var(--brand-text); + text-transform: lowercase; + } +} + +.comingSoon { + font-weight: 300; + font-size: 18px; + line-height: 34px; + color: var(--text-secondary); +} + +.settingsCaption { + font-size: 18px; + font-weight: 800; + line-height: 20px; + color: var(--text-secondary); + margin-bottom: 20px; +} + +.devider { + width: 100%; + border-bottom: solid 1px var(--subtile-devider); + margin-block: 32px; +} diff --git a/src/pages/Settings.tsx b/src/pages/Settings.tsx new file mode 100644 index 0000000..6db67bc --- /dev/null +++ b/src/pages/Settings.tsx @@ -0,0 +1,74 @@ +import { Component } from 'solid-js'; +import Branding from '../components/Branding/Branding'; +import styles from './Settings.module.scss'; + +import FeedSorter from '../components/FeedSorter/FeedSorter'; +import ThemeChooser from '../components/ThemeChooser/ThemeChooser'; +import Wormhole from '../components/Wormhole/Wormhole'; +import { useIntl } from '@cookbook/solid-intl'; +import SettingsZap from '../components/SettingsZap/SettingsZap'; +import Search from '../components/Search/Search'; +import SettingsNotifications from '../components/SettingsNotifications/SettingsNotifications'; +import { settings as t } from '../translations'; + +const Settings: Component = () => { + + const intl = useIntl(); + + return ( +
+ + + + + + + + +
+
+ {intl.formatMessage(t.title)} +
+
+
+ {intl.formatMessage(t.theme)} +
+ + + +
+ +
+ {intl.formatMessage(t.feeds)} +
+ +
+ +
+ +
+ +
+ {intl.formatMessage(t.feeds)} +
+ +
+ +
+ +
+ +
+ {intl.formatMessage(t.notifications.title)} +
+ +
+ +
+
+ ) +} + +export default Settings; diff --git a/src/pages/Thread.module.scss b/src/pages/Thread.module.scss new file mode 100644 index 0000000..90faedf --- /dev/null +++ b/src/pages/Thread.module.scss @@ -0,0 +1,61 @@ +.repliesHolder { + position: relative; + padding-bottom: 60px; +} +.noContent { + position: relative; + color: var(--text-secondary); + text-align: center; + margin-top: 80px; +} + +.border { + height: 36px; + padding: 1px; + background: var(--brand-gradient); + border-radius: 6px; + margin-left: 10px; + + input { + height: 34px; + font-size: 18px; + line-height: 20px; + margin: 0px; + border-radius: 6px; + border: none; + background-color: var(--background-site); + } +} + +.replyBox { + margin-top: 4px; + padding: 30px 22px; + background-color: var(--background-card); +} + +@media only screen and (max-width: 720px) { + .border { + height: 36px; + padding: 1px; + background: var(--brand-gradient); + border-radius: 6px; + margin-left: 10px; + + input { + height: 34px; + font-size: 18px; + line-height: 20px; + margin: 0px; + border-radius: 6px; + border: none; + background-color: var(--background-site); + } + } + + .replyBox { + margin-top: 4px; + padding: 30px 12px; + width: 100%; + background-color: var(--background-card); + } +} diff --git a/src/pages/Thread.tsx b/src/pages/Thread.tsx new file mode 100644 index 0000000..0990c0e --- /dev/null +++ b/src/pages/Thread.tsx @@ -0,0 +1,174 @@ +import { Component, createEffect, For, onCleanup, Show } from 'solid-js'; +import Note from '../components/Note/Note'; +import styles from './Thread.module.scss'; +import { useParams } from '@solidjs/router'; +import { PrimalNote } from '../types/primal'; +import NotePrimary from '../components/Note/NotePrimary/NotePrimary'; +import PeopleList from '../components/PeopleList/PeopleList'; +import PageNav from '../components/PageNav/PageNav'; +import ReplyToNote from '../components/ReplyToNote/ReplyToNote'; + +import Loader from '../components/Loader/Loader'; +import { nip19 } from 'nostr-tools'; +import { useThreadContext } from '../contexts/ThreadContext'; +import Wormhole from '../components/Wormhole/Wormhole'; +import { useAccountContext } from '../contexts/AccountContext'; +import { sortByRecency } from '../stores/note'; +import { scrollWindowTo } from '../lib/scroll'; +import { useIntl } from '@cookbook/solid-intl'; +import Search from '../components/Search/Search'; +import { thread as t } from '../translations'; + + +const Thread: Component = () => { + const account = useAccountContext(); + const params = useParams(); + const intl = useIntl(); + + const postId = () => { + if (params.postId.startsWith('note')) { + return params.postId; + } + + return nip19.noteEncode(params.postId); + }; + + const threadContext = useThreadContext(); + + const primaryNote = () => { + // const id = postId(); + // const savedNote = threadContext?.primaryNote; + + + // if (savedNote?.post.noteId === postId()) { + // return savedNote; + // } + + return threadContext?.notes.find(n => n.post.noteId === postId()); + }; + + const parentNotes = () => { + const note = primaryNote(); + + if (!note) { + return []; + } + + return sortByRecency( + threadContext?.notes.filter(n => + n.post.id !== note.post.id && n.post.created_at <= note.post.created_at, + ) || [], + true, + ); + }; + + const replyNotes = () => { + const note = primaryNote(); + + if (!note) { + return []; + } + + return threadContext?.notes.filter(n => + n.post.id !== note.post.id && n.post.created_at >= note.post.created_at, + ) || []; + }; + + const people = () => threadContext?.users || []; + const isFetching = () => threadContext?.isFetching; + + createEffect(() => { + threadContext?.actions.fetchNotes(postId()); + }); + + let observer: IntersectionObserver | undefined; + + createEffect(() => { + if (primaryNote() && !threadContext?.isFetching) { + const pn = document.getElementById('primary_note'); + + if (!pn) { + return; + } + + observer = new IntersectionObserver(entries => { + const rect = pn.getBoundingClientRect(); + entries.forEach((entry) => { + if (!entry.isIntersecting) { + scrollWindowTo(rect.top); + } + observer?.unobserve(pn); + }); + }); + + observer?.observe(pn); + } + }); + + onCleanup(() => { + const pn = document.getElementById('primary_note'); + + pn && observer?.unobserve(pn); + }); + + return ( +
+ + + + + + + + + + + + + + + {note => +
+ +
+ } +
+
+ + +
+ + + + +
+
+ +
+
} + > + + {note => +
+ +
+ } +
+ +
+ + ) +} + +export default Thread; diff --git a/src/services/StoreService.ts b/src/services/StoreService.ts new file mode 100644 index 0000000..8d218dc --- /dev/null +++ b/src/services/StoreService.ts @@ -0,0 +1,113 @@ +import { nip19 } from "nostr-tools"; +import { createStore } from "solid-js/store"; +import { APP_ID } from "../App"; +import { emptyPage, Kind } from "../constants"; +import { convertToNotes, sortingPlan } from "../stores/note"; +import { FeedPage, NostrEventContent, NostrMentionContent, NostrNoteActionsContent, NostrNoteContent, NostrStatsContent, NostrUserContent, NoteActions, PrimalNote } from "../types/primal"; + +type FeedStore = { + lastNote?: PrimalNote, + notes: PrimalNote[], + isFetching: boolean, +} + +type PrimalStore = { + page: Record, + feed: Record, +}; + + +export const [store, updateStore] = createStore({ + page: {}, + feed: {}, +}); + +export const getStoreKey = (subId: string) => { + return subId.replace(APP_ID, ''); +}; + +export const updatePage = (subId: string, content: NostrEventContent) => { + + const storeKey = getStoreKey(subId); + const feed = store.feed[storeKey]; + + if (content.kind === Kind.Metadata) { + const user = content as NostrUserContent; + + + updateStore('page', storeKey, 'users', + (usrs) => ({ ...usrs, [user.pubkey]: { ...user } }) + ); + return; + } + + if ([Kind.Text, Kind.Repost].includes(content.kind)) { + const message = content as NostrNoteContent; + const messageId = nip19.noteEncode(message.id); + + const isLastNote = message.kind === Kind.Text ? + feed.lastNote?.post?.noteId === messageId : + feed.lastNote?.repost?.note.noteId === messageId; + + if (!isLastNote) { + updateStore('page', storeKey, 'messages', + (msgs) => [ ...msgs, { ...message }] + ); + } + + return; + } + + if (content.kind === Kind.NoteStats) { + const statistic = content as NostrStatsContent; + const stat = JSON.parse(statistic.content); + + updateStore('page', storeKey, 'postStats', + (stats) => ({ ...stats, [stat.event_id]: { ...stat } }) + ); + return; + } + + if (content.kind === Kind.Mentions) { + const mentionContent = content as NostrMentionContent; + const mention = JSON.parse(mentionContent.content); + + updateStore('page', storeKey, 'mentions', + (mentions) => ({ ...mentions, [mention.id]: { ...mention } }) + ); + return; + } + + if (content.kind === Kind.NoteActions) { + const noteActionContent = content as NostrNoteActionsContent; + const noteActions = JSON.parse(noteActionContent.content) as NoteActions; + + updateStore('page', storeKey, 'noteActions', + (actions) => ({ ...actions, [noteActions.event_id]: { ...noteActions } }) + ); + return; + } +}; + +export const savePage = (subId: string, sortBy = 'latest') => { + const storeKey = getStoreKey(subId); + const sortingFunction = sortingPlan(sortBy); + + const newPosts = sortingFunction(convertToNotes(store.page[storeKey])); + + saveNotes(newPosts, storeKey); +}; + +export const saveNotes = (notes: PrimalNote[], subId: string) => { + const storeKey = getStoreKey(subId); + + updateStore('feed', storeKey, 'notes', (nts) => [ ...nts, ...notes ]); + updateStore('feed', storeKey, 'isFetching', () => false); +}; + + +export const clearPage = (subId: string) => { + const storeKey = getStoreKey(subId); + + updateStore('page', storeKey, () => ({ ...emptyPage })); +}; diff --git a/src/sockets.tsx b/src/sockets.tsx new file mode 100644 index 0000000..62446af --- /dev/null +++ b/src/sockets.tsx @@ -0,0 +1,102 @@ +import { createSignal } from "solid-js"; +import { NostrEvent, NostrEOSE, NostrEventType, NostrEventContent } from "./types/primal"; + +export const [socket, setSocket] = createSignal(); + +export const [isConnected, setConnected] = createSignal(false); + +export const isNotConnected = () => !isConnected(); + +const onOpen = () => { + setConnected(true); +} + +const onClose = () => { + setConnected(false); + + socket()?.removeEventListener('open', onOpen); + socket()?.removeEventListener('close', onClose); + socket()?.removeEventListener('error', onError); + + setTimeout(() => { + connect(); + }, 200); +} + +const onError = (error: Event) => { + console.log("ws error: ", error); +}; + +export const connect = () => { + if (isNotConnected()) { + const cacheServer = localStorage.getItem('cacheServer') ?? + 'wss://cache3.primal.net/cache17'; + + setSocket(new WebSocket(cacheServer)); + console.log('CACHE SOCKET: ', socket()); + + socket()?.addEventListener('open', onOpen); + socket()?.addEventListener('close', onClose); + socket()?.addEventListener('error', onError); + } +}; + +export const disconnect = () => { + socket()?.close(); +}; + +export const reset = () => { + disconnect(); + setTimeout(connect, 1000); +}; + +export const sendMessage = (message: string) => { + isConnected() && socket()?.send(message); +} + +export const refreshSocketListeners = ( + ws: WebSocket | undefined, + listeners: Record any>, + ) => { + + if (!ws) { + return; + } + + Object.keys(listeners).forEach((event: string) => { + ws.removeEventListener(event, listeners[event]); + ws.addEventListener(event, listeners[event]); + }); +}; + +export const removeSocketListeners = ( + ws: WebSocket | undefined, + listeners: Record any>, + ) => { + + if (!ws) { + return; + } + + Object.keys(listeners).forEach((event: string) => { + ws.removeEventListener(event, listeners[event]); + }); +}; + +export const subscribeTo = (subId: string, cb: (type: NostrEventType, subId: string, content?: NostrEventContent) => void ) => { + const listener = (event: MessageEvent) => { + const message: NostrEvent | NostrEOSE = JSON.parse(event.data); + const [type, subscriptionId, content] = message; + + if (subId === subscriptionId) { + cb(type, subscriptionId, content); + } + + }; + + socket()?.addEventListener('message', listener); + + return () => { + socket()?.removeEventListener('message', listener); + }; +}; diff --git a/src/stores/note.ts b/src/stores/note.ts new file mode 100644 index 0000000..0d800dd --- /dev/null +++ b/src/stores/note.ts @@ -0,0 +1,349 @@ +import { nip19 } from "nostr-tools"; +import { Kind } from "../constants"; +import { hexToNpub } from "../lib/keys"; +import { sanitize } from "../lib/notes"; +import { RepostInfo, NostrNoteContent, FeedPage, PrimalNote, PrimalRepost, NostrEventContent, NostrEOSE, NostrEvent, PrimalUser } from "../types/primal"; +import { convertToUser, emptyUser } from "./profile"; + + +export const getRepostInfo: RepostInfo = (page, message) => { + const user = page?.users[message.pubkey]; + const userMeta = JSON.parse(user?.content || '{}'); + const stat = page?.postStats[message.id]; + + + const noActions = { + event_id: message.id, + liked: false, + replied: false, + reposted: false, + zapped: false, + }; + + return { + user: { + id: user?.id || '', + pubkey: user?.pubkey || message.pubkey, + npub: hexToNpub(user?.pubkey || message.pubkey), + name: (userMeta.name || user?.pubkey) as string, + about: (userMeta.about || '') as string, + picture: (userMeta.picture || '') as string, + nip05: (userMeta.nip05 || '') as string, + banner: (userMeta.banner || '') as string, + displayName: (userMeta.display_name || '') as string, + location: (userMeta.location || '') as string, + lud06: (userMeta.lud06 || '') as string, + lud16: (userMeta.lud16 || '') as string, + website: (userMeta.website || '') as string, + tags: user?.tags || [], + }, + note: { + id: message.id, + pubkey: message.pubkey, + created_at: message.created_at || 0, + tags: message.tags, + content: sanitize(message.content), + sig: message.sig, + likes: stat?.likes || 0, + mentions: stat?.mentions || 0, + reposts: stat?.reposts || 0, + replies: stat?.replies || 0, + zaps: stat?.zaps || 0, + score: stat?.score || 0, + score24h: stat?.score24h || 0, + satszapped: stat?.satszapped || 0, + noteId: nip19.noteEncode(message.id), + noteActions: (page.noteActions && page.noteActions[message.id]) || noActions, + }, + } +}; + +export const parseEmptyReposts = (page: FeedPage) => { + let reposts: Record = {}; + + page.messages.forEach(message => { + if (message.kind === 6 && message.content.length === 0) { + const tag = message.tags.find(t => t[0] === 'e'); + if (tag) { + reposts[tag[1]] = message.id; + } + } + }); + + return reposts; +}; + +const parseKind6 = (message: NostrNoteContent) => { + try { + return JSON.parse(message.content); + } catch (e) { + return { + kind: 1, + content: '', + id: message.id, + created_at: message.created_at, + pubkey: message.pubkey, + sig: message.sig, + tags: message.tags, + } + } +}; + +// const getNoteReferences = (message: NostrNoteContent) => { +// const regex = /\#\[([0-9]*)\]/g; +// let refs = []; +// let match; + +// while((match = regex.exec(message.content)) !== null) { +// refs.push(match[1]); +// } + +// return refs.reduce((acc, ref) => { +// const tag = message.tags[parseInt(ref)] || []; + +// return tag[0] === 'e' ? [...acc, tag[1]] : acc; +// }, []); +// }; + +// const getUserReferences = (message: NostrNoteContent) => { +// const regex = /\#\[([0-9]*)\]/g; +// let refs = []; +// let match; + +// while((match = regex.exec(message.content)) !== null) { +// refs.push(match[1]); +// } + +// return refs.reduce((acc, ref) => { +// const tag: string[] = message.tags[parseInt(ref)] || []; + +// return tag[0] === 'p' ? [...acc, tag[1]] : acc; +// }, []); +// }; + +type ConvertToNotes = (page: FeedPage | undefined) => PrimalNote[]; + +export const convertToNotes: ConvertToNotes = (page) => { + + if (page === undefined) { + return []; + } + + const mentions = page.mentions || {}; + + return page.messages.map((message) => { + const msg: NostrNoteContent = message.kind === Kind.Repost ? parseKind6(message) : message; + + const user = page?.users[msg.pubkey]; + const stat = page?.postStats[msg.id]; + + const userMeta = JSON.parse(user?.content || '{}'); + + const mentionIds = Object.keys(mentions) //message.tags.reduce((acc, t) => t[0] === 'e' ? [...acc, t[1]] : acc, []); + const userMentionIds = message.tags.reduce((acc, t) => t[0] === 'p' ? [...acc, t[1]] : acc, []); + + let mentionedNotes: Record = {}; + let mentionedUsers: Record = {}; + + if (mentionIds.length > 0) { + for (let i = 0;i 0) { + for (let i = 0;i { + + const aData: Record = a.repost ? a.repost.note : a.post; + const bData: Record = b.repost ? b.repost.note : b.post; + + return bData[property] - aData[property]; +}; + +export const sortByRecency = (posts: PrimalNote[], reverse = false) => { + return posts.sort((a: PrimalNote, b: PrimalNote) => { + const order = sortBy(a, b, 'created_at'); + + return reverse ? -1 * order : order; + }); +}; + +export const sortByScore24h = (posts: PrimalNote[], reverse = false) => { + return posts.sort((a: PrimalNote, b: PrimalNote) => { + const order = sortBy(a, b, 'score24h'); + + return reverse ? -1 * order : order; + }); +}; + +export const sortByScore = (posts: PrimalNote[], reverse = false) => { + return posts.sort((a: PrimalNote, b: PrimalNote) => { + const order = sortBy(a, b, 'score'); + + return reverse ? -1 * order : order; + }); +}; + +export const sortByZapped = (posts: PrimalNote[], reverse = false) => { + return posts.sort((a: PrimalNote, b: PrimalNote) => { + const order = sortBy(a, b, 'satszapped'); + + return reverse ? -1 * order : order; + }); +}; + +export const sortingPlan = (topic: string = '') => { + const sortingFunctions: Record = { + trending: sortByScore24h, + popular: sortByScore, + latest: sortByRecency, + mostzapped: sortByZapped, + mostzapped4h: sortByZapped, + } + + const plan = topic || 'latest'; + + return sortingFunctions[plan] || sortingFunctions['latest']; +}; + + +export const paginationPlan = (criteria: string) => { + const pagCriteria: Record = { + trending: 'score24h', + popular: 'score', + latest: 'created_at', + mostzapped: 'satszapped', + mostzapped4h: 'satszapped', + } + + const plan = criteria || 'latest'; + + return pagCriteria[plan] || pagCriteria['latest']; +} + +type NoteStore = { + notes: PrimalNote[], + page: FeedPage, + lastNote: PrimalNote | undefined, + reposts: Record | undefined, +} + +export const referencesToTags = (value: string) => { + const regex = + /\bnostr:((note|npub|nevent|nprofile)1\w+)\b|#\[(\d+)\]/g; + + let refs: string[] = []; + let tags: string[][] = []; + let match; + + while((match = regex.exec(value)) !== null) { + refs.push(match[0]); + } + + refs.forEach((ref) => { + const decoded = nip19.decode(ref.split('nostr:')[1]); + + if (decoded.type === 'npub') { + tags.push(['p', decoded.data, '', 'mention']) + return; + } + + if (decoded.type === 'nprofile') { + const relay = decoded.data.relays ? decoded.data.relays[0] : ''; + tags.push(['p', decoded.data.pubkey, relay, 'mention']); + return; + } + + if (decoded.type === 'note') { + tags.push(['e', decoded.data, '', 'mention']); + return; + } + + if (decoded.type === 'nevent') { + const relay = decoded.data.relays ? decoded.data.relays[0] : ''; + tags.push(['e', decoded.data.id, relay, 'mention']); + return; + } + }); + + return tags; + +}; diff --git a/src/stores/profile.ts b/src/stores/profile.ts new file mode 100644 index 0000000..87253ac --- /dev/null +++ b/src/stores/profile.ts @@ -0,0 +1,96 @@ +import { hexToNpub } from "../lib/keys"; +import { NostrUserContent, PrimalUser } from "../types/primal"; + +export const truncateNpub = (npub: string) => { + if (npub.length < 24) { + return npub; + } + return `${npub.slice(0, 15)}..${npub.slice(-10)}`; +}; + +export const truncateName = (name: string, limit = 20) => { + if (name.length < limit) { + return name; + } + return `${name.slice(0, limit)}...`; +}; + +export const convertToUser: (user: NostrUserContent) => PrimalUser = (user: NostrUserContent) => { + const userMeta = JSON.parse(user.content || '{}'); + + return { + id: user.id, + pubkey: user.pubkey, + tags: user.tags, + npub: hexToNpub(user.pubkey), + name: (userMeta.name || '') as string, + about: (userMeta.about || '') as string, + picture: (userMeta.picture || '') as string, + nip05: (userMeta.nip05 || '') as string, + banner: (userMeta.banner || '') as string, + displayName: (userMeta.display_name || '') as string, + location: (userMeta.location || '') as string, + lud06: (userMeta.lud06 || '') as string, + lud16: (userMeta.lud16 || '') as string, + website: (userMeta.website || '') as string, + }; +} + +export const emptyUser = (pubkey: string) => { + return { + id: '', + pubkey, + tags: [], + npub: hexToNpub(pubkey), + name: '', + about: '', + picture: '', + nip05: '', + banner: '', + displayName: '', + location: '', + lud06: '', + lud16: '', + website: '', + } as PrimalUser; +}; + +export const userName = (user: PrimalUser | undefined) => { + if (!user) { + return ''; + } + const name = user.name || + user.display_name || + user.displayName || + user.npub; + + return name ? + truncateName(name) : + truncateNpub(hexToNpub(user.pubkey) || ''); +}; + +export const authorName = (user: PrimalUser | undefined) => { + if (!user) { + return ''; + } + const name = user.display_name || + user.displayName || + user.name || + user.npub; + + return name ? + truncateName(name) : + truncateNpub(hexToNpub(user.pubkey) || ''); +}; + +export const nip05Verification = (user: PrimalUser | undefined) => { + if (!user) { + return ''; + } + + if (user.nip05.startsWith('_@')) { + return user.nip05.slice(2); + } + + return user.nip05; +}; diff --git a/src/stores/trending.ts b/src/stores/trending.ts new file mode 100644 index 0000000..511c61e --- /dev/null +++ b/src/stores/trending.ts @@ -0,0 +1,65 @@ +import { createStore, SetStoreFunction } from "solid-js/store"; +import { sortByScore24h, convertToNotes } from "../stores/note"; +import { NostrNoteContent, NostrUserContent, NostrStatsContent, FeedPage, PrimalNote, NostrEventContent, Kind } from "../types/primal"; + +export type TrendingNotesData = FeedPage & { notes: PrimalNote[]}; + +export const emptyNotes: TrendingNotesData = { + messages: [], + users: {}, + notes: [], + postStats: {}, +}; + +export const [trendingNotes, setTrendingNotes] = + createStore(emptyNotes); + +const proccessNote = (post: NostrNoteContent) => { + setTrendingNotes('messages', (msgs) => [ ...msgs, post]); +}; + +const proccessUser = (user: NostrUserContent) => { + setTrendingNotes('users', (users) => ({ ...users, [user.pubkey]: user})) +}; + +const proccessStat = (stat: NostrStatsContent) => { + const content = JSON.parse(stat.content); + setTrendingNotes('postStats', (stats) => ({ ...stats, [content.event_id]: content })) +}; + +export const processTrendingNotes = (type: string, content: NostrEventContent | undefined) => { + if (type === 'EOSE') { + const newNotes = sortByScore24h(convertToNotes(trendingNotes)); + + setTrendingNotes('notes', () => [...newNotes]); + + return; + } + + if (type === 'EVENT') { + if (content && content.kind === Kind.Metadata) { + proccessUser(content); + } + if (content && content.kind === Kind.Text) { + proccessNote(content); + } + if (content && content.kind === Kind.Repost) { + proccessNote(content); + } + if (content && content.kind === Kind.NoteStats) { + proccessStat(content); + } + } +}; + +export type TrendingNotesStore = { + data: TrendingNotesData, + setTrendingNotes: SetStoreFunction, + processTrendingNotes: (type: string, content: NostrEventContent | undefined) => void, +} + +export default { + data: trendingNotes, + setTrendingNotes, + processTrendingNotes, +}; diff --git a/src/translations.ts b/src/translations.ts new file mode 100644 index 0000000..307affb --- /dev/null +++ b/src/translations.ts @@ -0,0 +1,773 @@ +import { MessageDescriptor } from "@cookbook/solid-intl"; +import { NotificationType } from "./constants"; +import { ScopeDescriptor } from "./types/primal"; + +export const account = { + follow: { + id: 'actions.follow', + defaultMessage: 'follow', + description: 'Follow button label', + }, + unfollow: { + id: 'actions.unfollow', + defaultMessage: 'unfollow', + description: 'Unfollow button label', + }, + needToLogin: { + id: 'account.needToLogin', + defaultMessage: 'You need to be signed in to perform this action', + description: 'Message to user that an action cannot be preformed without a public key', + }, +}; + +export const actions = { + cancel: { + id: 'actions.cancel', + defaultMessage: 'cancel', + description: 'Cancel action, button label', + }, + addFeedToHome: { + id: 'actions.addFeedToHome', + defaultMessage: 'add this feed to my home page', + description: 'Add feed to home, button label', + }, + addFeedToHomeNamed: { + id: 'actions.addFeedToHomeNamed', + defaultMessage: 'add {name} feed to home page', + description: 'Add named feed to home, button label', + }, + disabledAddFeedToHome: { + id: 'actions.disabledHomeFeedAdd', + defaultMessage: 'Available on your home page', + description: 'Add feed to home label, when feed is already added', + }, + removeFromHomeFeedNamed: { + id: 'actions.removeFromHomeFeedNamed', + defaultMessage: 'remove {name} feed from your home page', + description: 'Remove named feed from home, button label', + }, + noteCopyNostrLink: { + id: 'actions.noteCopyNostrLink', + defaultMessage: 'Copy Nostr link', + description: 'Label for the copy Nostr note link context menu item', + }, + noteCopyPrimalLink: { + id: 'actions.noteCopyPrimalLink', + defaultMessage: 'Copy Primal link', + description: 'Label for the copy Primal note link context menu item', + }, + notePostNew: { + id: 'actions.notePostNew', + defaultMessage: 'post', + description: 'Send new note, button label', + }, + noteReply: { + id: 'actions.noteReply', + defaultMessage: 'reply to {name}', + description: 'Reply to button label', + }, + sendDirectMessage: { + id: 'actions.sendDirectMessage', + defaultMessage: 'send', + description: 'Send direct message action, button label', + }, +}; + +export const branding = { + id: 'branding', + defaultMessage: 'Primal', + description: 'Brand name', +}; + +export const exploreSidebarCaption = { + id: 'explore.sidebar.caption', + defaultMessage: 'trending users', + description: 'Caption for the explore page sidebar showing a list of trending users', +}; + +export const explore = { + genericCaption: { + id: 'explore.genericCaption', + defaultMessage: 'explore nostr', + description: 'Generic caption for the explore page', + }, + title: { + id: 'explore.title', + defaultMessage: '{timeframe}: {scope}', + description: 'Title of the explore page', + }, + statDisplay: { + users: { + id: 'explore.stats.users', + defaultMessage:'Users', + description: 'Label for number of users stats', + }, + pubkeys: { + id: 'explore.stats.pubkeys', + defaultMessage: 'Public Keys', + description: 'Label for number of pubkeys stats', + }, + zaps: { + id: 'explore.stats.zaps', + defaultMessage: 'Zaps', + description: 'Label for number of zaps stats', + }, + btcZapped: { + id: 'explore.stats.btcZapped', + defaultMessage: 'BTC Zapped', + description: 'Label for number of zapped bitcoins stats', + }, + pubnotes: { + id: 'explore.stats.pubnotes', + defaultMessage: 'Public Notes', + description: 'Label for number of public notes stats', + }, + reposts: { + id: 'explore.stats.reposts', + defaultMessage: 'Reposts', + description: 'Label for number of repost stats', + }, + reactions: { + id: 'explore.stats.reactions', + defaultMessage: 'Reactions', + description: 'Label for number of reactions stats', + }, + any: { + id: 'explore.stats.any', + defaultMessage: 'All Events', + description: 'Label for number of all stats', + }, + } +}; + +export const feedNewPosts = { + id: 'feed.newPosts', + defaultMessage: `{number, plural, + =0 {} + one {# new post} + =100 {99+ new posts} + other {# new posts}}`, + description: 'Label for a button to load new posts', +}; + +export const feedback = { + dropzone: { + id: 'feedback.dropzone', + defaultMessage: 'drop file to upload', + description: 'Label accompanying the draging file' + }, + uploading: { + id: 'feedback.uploading', + defaultMessage: 'uploading...', + description: 'Label accompanying the uploading spinner' + }, +}; + +export const messages = { + title: { + id: 'messages.title', + defaultMessage: 'Messages', + description: 'Title of messages page', + }, + follows: { + id: 'messages.follows', + defaultMessage: 'follows', + description: 'DM relation selection label for follows', + }, + other: { + id: 'messages.other', + defaultMessage: 'other', + description: 'DM relation selection label for other', + }, + markAsRead: { + id: 'messages.markAsRead', + defaultMessage: 'Mark All Read', + description: 'DM mark as read label', + }, +}; + +export const navBar = { + home: { + id: 'navbar.home', + defaultMessage: 'Home', + description: 'Label for the nav bar item link to Home page', + }, + explore: { + id: 'navbar.explore', + defaultMessage: 'Explore', + description: 'Label for the nav bar item link to Explore page', + }, + messages: { + id: 'navbar.messages', + defaultMessage: 'Messages', + description: 'Label for the nav bar item link to Messages page', + }, + notifications: { + id: 'navbar.notifications', + defaultMessage: 'Notifications', + description: 'Label for the nav bar item link to Notifications page', + }, + downloads: { + id: 'navbar.downloads', + defaultMessage: 'Downloads', + description: 'Label for the nav bar item link to Downloads page', + }, + settings: { + id: 'navbar.settings', + defaultMessage: 'Settings', + description: 'Label for the nav bar item link to Settings page', + }, + help: { + id: 'navbar.help', + defaultMessage: 'Help', + description: 'Label for the nav bar item link to Help page', + }, +}; + +export const note = { + newPreview: { + id: 'note.newPreview', + defaultMessage: 'Note preview', + description: 'Caption for preview when creating a new note' + }, + mentionIndication: { + id: 'note.mentionIndication', + defaultMessage: '\[post by {name}\]', + description: 'Label indicating that a note has been metioned in the small note display' + }, + reposted: { + id: 'note.reposted', + defaultMessage: 'Reposted', + description: 'Label indicating that the note is a repost', + }, +}; + +export const notificationTypeTranslations: Record = { + [NotificationType.NEW_USER_FOLLOWED_YOU]: 'followed you', + [NotificationType.USER_UNFOLLOWED_YOU]: 'unfollowed you', + + [NotificationType.YOUR_POST_WAS_ZAPPED]: 'zapped your post', + [NotificationType.YOUR_POST_WAS_LIKED]: 'liked your post', + [NotificationType.YOUR_POST_WAS_REPOSTED]: 'reposted your post', + [NotificationType.YOUR_POST_WAS_REPLIED_TO]: 'replied to your post', + + [NotificationType.YOU_WERE_MENTIONED_IN_POST]: 'mentioned you in a post', + [NotificationType.YOUR_POST_WAS_MENTIONED_IN_POST]: 'mentioned your post', + + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_ZAPPED]: 'zapped a post you were mentioned in', + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_LIKED]: 'liked a post you were mentioned in', + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_REPOSTED]: 'reposted a post you were mentioned in', + [NotificationType.POST_YOU_WERE_MENTIONED_IN_WAS_REPLIED_TO]: 'replied to a post you were mentioned in', + + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_ZAPPED]: 'zapped a post your post was mentioned in', + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_LIKED]: 'liked a post your post was mentioned in', + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPOSTED]: 'reposted a post your post was mentioned in', + [NotificationType.POST_YOUR_POST_WAS_MENTIONED_IN_WAS_REPLIED_TO]: 'replied to a post your post was mentioned in', +} + +export const notificationsNew: Record = Object.values(NotificationType).reduce((acc, type) => ({ + ...acc, + [type]: { + id: `notifications.new.${type}`, + defaultMessage: `{number, plural, + =0 {} + one {and # other} + other {and # others}} + ${notificationTypeTranslations[type]}`, + description: `New Notifiaction label for notifications of type ${type}`, + }, +}), {}); + +export const notificationsOld: Record = Object.values(NotificationType).reduce((acc, type) => ({ + ...acc, + [type]: { + id: `notifications.old.${type}`, + defaultMessage: `${notificationTypeTranslations[type]}`, + description: `Old Notifiaction label for notifications of type ${type}`, + }, +}), {}); + +export const notificationsSidebar = { + activities: { + id: 'notifications.sidebar.activities', + defaultMessage: 'Reactions', + description: 'Sidebar activities stats caption on the notification page', + }, + heading: { + id: 'notificationsSidebar.heading', + defaultMessage: 'Summary', + description: 'Sidebar caption on the notification page', + }, + empty: { + id: 'notificationsSidebar.empty', + defaultMessage: 'No new notifications', + description: 'Sidebar caption indicating no new notifications', + }, + followers: { + id: 'notificationsSidebar.followers', + defaultMessage: 'Followers', + description: 'Sidebar follower stats caption on the notification page', + }, + gainedFollowers: { + id: 'notificationsSidebar.gainedFollowers', + defaultMessage: `new {number, plural, + =0 {} + one {follower} + other {followers}}`, + description: 'Sidebar new follower stats description on the notification page', + }, + lostFollowers: { + id: 'notificationsSidebar.lostFollowers', + defaultMessage: `lost {number, plural, + =0 {} + one {follower} + other {followers}}`, + description: 'Sidebar lost follwers stats description on the notification page', + }, + likes: { + id: 'notifications.sidebar.likes', + defaultMessage: `{number, plural, + =0 {} + one {like} + other {likes}}`, + description: 'Sidebar likes stats caption on the notification page', + }, + mentions: { + id: 'notifications.sidebar.mentions', + defaultMessage: 'Mentions', + description: 'Sidebar mentions stats caption on the notification page', + }, + mentionsYou: { + id: 'notifications.sidebar.mentionsYou', + defaultMessage: `{number, plural, + =0 {} + one {mention} + other {mentions}} of you`, + description: 'Sidebar mentions you stats description on the notification page', + }, + mentionsYourPost: { + id: 'notifications.sidebar.mentionsYourPost', + defaultMessage: `{number, plural, + =0 {} + one {mention of your post} + other {mentions of your posts}}`, + description: 'Sidebar mentions your post stats description on the notification page', + }, + replies: { + id: 'notifications.sidebar.replies', + defaultMessage: `{number, plural, + =0 {} + one {reply} + other {replies}}`, + description: 'Sidebar replies stats caption on the notification page', + }, + reposts: { + id: 'notifications.sidebar.reposts', + defaultMessage: `{number, plural, + =0 {} + one {repost} + other {reposts}}`, + description: 'Sidebar reposts stats caption on the notification page', + }, + other: { + id: 'notifications.sidebar.other', + defaultMessage: 'Other', + description: 'Sidebar other stats caption on the notification page', + }, + zaps: { + id: 'notificationsSidebar.zaps', + defaultMessage: 'Zaps', + description: 'Sidebar zaps stats caption on the notification page', + }, + zapNumber: { + id: 'notificationsSidebar.zapNumber', + defaultMessage: `{number, plural, + =0 {} + one {zap} + other {zaps}}`, + description: 'Sidebar zaps stats description on the notification page', + }, + statsNumber: { + id: 'notificationsSidebar.statsNumber', + defaultMessage: `{number, plural, + =0 {} + one {sat} + other {sats}}`, + description: 'Sidebar sats stats description on the notification page', + }, +}; + +export const notifications = { + title: { + id: 'pages.notifications.title', + defaultMessage: 'Notifications', + description: 'Title of the notifications page', + }, + newNotifs: { + id: 'notification.newNotifs', + defaultMessage: `{number, plural, + =0 {} + one {# new notification} + =100 {99+ new notifications} + other {# new notifications}}`, + description: 'Label for a button to load new notifications', + }, +}; + +export const placeholders = { + comingSoon: { + id: 'placeholders.comingSoon', + defaultMessage: 'Coming soon', + description: 'Placholder text for missing content', + }, + endOfFeed: { + id: 'placeholders.endOfFeed', + defaultMessage: 'Your reached the end. You are a quick reader', + description: 'Message displayed when user reaches the end of the feed', + }, + guestUserGreeting: { + id: 'placeholders.guestUserGreeting', + defaultMessage: 'Welcome to nostr!', + description: 'Header placeholder for guest user', + }, + noteCallToAction: { + id: 'placeholders.callToAction.note', + defaultMessage: 'say something on nostr...', + description: 'Placeholder for new note call-to-action', + }, + pageWIPTitle: { + id: 'pages.wip.title', + defaultMessage: '{title}', + description: 'Title of page under construction', + }, + welcomeMessage: { + id: 'placeholders.welcomeMessage', + defaultMessage: 'Welcome to nostr!', + description: 'Default welcome message', + }, + findUser: { + id: 'placeholders.findUser', + defaultMessage: 'find user', + description: 'Find user input placeholder', + }, + findUsers: { + id: 'placeholders.findUsers', + defaultMessage: 'find users', + description: 'Find users input placeholder', + }, + search: { + id: 'placeholders.search', + defaultMessage: 'search', + description: 'Search input placeholder', + }, + selectFeed: { + id: 'placeholders.selectFeed', + defaultMessage: 'Select feed', + description: 'Placeholder for feed selection', + }, + pageNotFound: { + id: 'placeholders.pageNotFound', + defaultMessage: 'Page not found', + description: 'Placholder text for missing page', + }, +}; + +export const profile = { + sidebarCaption: { + id: 'profile.sidebar.caption', + defaultMessage: 'Popular posts', + description: 'Caption for the profile page sidebar showing a list of trending notes by the profile', + }, + sidebarNoNotes: { + id: 'profile.sidebar.noNotes', + defaultMessage: 'No trending posts', + description: 'Placeholde for profile sidebar when the profile is missing trending notes', + }, + title: { + id: 'profile.title', + defaultMessage: '{name} - Nostr Profile', + description: 'Page title for Profile page' + }, + followsYou: { + id: 'profile.followsYou', + defaultMessage: 'Follows you', + description: 'Label indicating that a profile is following your profile', + }, + jointDate: { + id: 'profile.joinDate', + defaultMessage: 'Joined Nostr on {date}', + description: 'Label indicating when the profile joined Nostr (oldest event)', + }, + stats: { + follow: { + id: 'profile.followStats', + defaultMessage: 'Following', + description: 'Label for following profile stat', + }, + followers: { + id: 'profile.stats.followers', + defaultMessage: 'Followers', + description: 'Label for followers profile stat', + }, + notes: { + id: 'profile.stats.notes', + defaultMessage: 'Posts', + description: 'Label for notes profile stat', + }, + }, +}; + +export const search = { + followers: { + id: 'search.followers', + defaultMessage: 'followers', + description: 'Followers label for user search results', + }, + invalid: { + id: 'search.invalid', + defaultMessage: 'Please enter search term.', + description: 'Alert letting the user know that the search term is empty', + }, + emptyQueryResult: { + id: 'search.emptyQueryResult', + defaultMessage: 'type to', + description: 'Label shown is search resuls when no term is provided', + }, + searchNostr: { + id: 'search.searchNostr', + defaultMessage: 'search nostr', + description: 'Label explaining full search action', + }, + sidebarCaption: { + id: 'search.sidebarCaption', + defaultMessage: 'Users found', + description: 'Caption for the search page sidebar showing a list of users', + }, + feedLabel: { + id: 'search.feedLabel', + defaultMessage: 'Search: {query}', + description: 'Label for a search results feed', + }, + title: { + id: 'search.title', + defaultMessage: 'search for "{query}"', + description: 'Title of the Search page', + }, + noResults: { + id: 'search.noResults', + defaultMessage: 'No results found', + description: 'Message shown when no search results were found' + }, +}; + +export const settings = { + title: { + id: 'settings.title', + defaultMessage: 'Settings', + description: 'Title of the settings page', + }, + theme: { + id: 'settings.sections.theme', + defaultMessage: 'Theme', + description: 'Title of the theme section on the settings page', + }, + feeds: { + id: 'settings.sections.feeds', + defaultMessage: 'Home page feeds', + description: 'Title of the feeds section on the settings page', + }, + zaps: { + id: 'settings.sections.zaps', + defaultMessage: 'Zaps', + description: 'Title of the zaps section on the settings page', + }, + notifications: { + title: { + id: 'pages.settings.sections.notifications', + defaultMessage: 'Notifications', + description: 'Title of the notifications section on the settings page', + }, + core: { + id: 'settings.sections.notifications.core', + defaultMessage: 'Core notifications:', + description: 'Title of the notification settings sub-section for core notifications', + }, + yourMentions: { + id: 'settings.sections.notifications.yourMentions', + defaultMessage: 'A post you were mentioned in was:', + description: 'Title of the notification settings sub-section for posts you were mentioned in', + }, + yourPostMentions: { + id: 'settings.sections.notifications.yourPostMentions', + defaultMessage: 'A post your post was mentioned in was:', + description: 'Title of the notification settings sub-section for posts your post was mentioned in', + }, + } +}; + +export const scopeDescriptors: Record = { + follows: { + caption: { + id: 'explore.scopes.follows.caption', + defaultMessage: 'Follows', + description: 'Caption for the follows scope', + }, + label: { + id: 'explore.scopes.follows.label', + defaultMessage: 'my follows', + description: 'Label for the follows scope', + }, + description: { + id: 'explore.scopes.follows.description', + defaultMessage: 'accounts you follow', + description: 'Description of the follows scope description', + }, + }, + tribe: { + caption: { + id: 'explore.scopes.tribe.caption', + defaultMessage: 'Tribe', + description: 'Caption for the tribe scope', + }, + label: { + id: 'explore.scopes.tribe.label', + defaultMessage: 'my tribe', + description: 'Label for the tribe scope', + }, + description: { + id: 'explore.scopes.tribe.description', + defaultMessage: 'accounts you follow + your followers', + description: 'Description of the tribe scope description', + }, + }, + network: { + caption: { + id: 'explore.scopes.network.caption', + defaultMessage: 'Network', + description: 'Caption for the network scope', + }, + label: { + id: 'explore.scopes.network.label', + defaultMessage: 'my network', + description: 'Label for the network scope', + }, + description: { + id: 'explore.scopes.network.description', + defaultMessage: 'accounts you follow + everyone they follow', + description: 'Description of the network scope description', + }, + }, + global: { + caption: { + id: 'explore.scopes.global.caption', + defaultMessage: 'Global', + description: 'Caption for the global scope', + }, + label: { + id: 'explore.scopes.global.label', + defaultMessage: 'global', + description: 'Label for the global scope', + }, + description: { + id: 'explore.scopes.global.description', + defaultMessage: 'all accounts on nostr', + description: 'Description of the global scope description', + }, + }, +}; + +export const timeframeDescriptors: Record = { + latest: { + id: 'explore.timeframes.latest.caption', + defaultMessage: 'latest', + description: 'Caption for the latest timeframe', + }, + trending: { + id: 'explore.timeframes.trending.caption', + defaultMessage: 'trending', + description: 'Caption for the trending timeframe', + }, + popular: { + id: 'explore.timeframes.popular.caption', + defaultMessage: 'popular', + description: 'Caption for the popular timeframe', + }, + mostzapped: { + id: 'explore.timeframes.mostzapped.caption', + defaultMessage: 'zapped', + description: 'Caption for the mostzapped timeframe', + }, +}; + +export const toastZapFail = { + id: 'toast.zapFail', + defaultMessage: 'We were unable to send this Zap', + description: 'Toast message indicating failed zap', +}; + +export const thread = { + sidebar: { + id: 'thread.sidebar.title', + defaultMessage: 'People in this thread', + description: 'Title of the Thread page sidebar', + }, +}; + +export const toast = { + addFeedToHomeSuccess: { + id: 'toasts.addFeedToHome.success', + defaultMessage: '"{name}" has been added to your home page', + description: 'Toast message confirming successfull adding of the feed to home to the list of available feeds', + }, + fileTypeUpsupported: { + id: 'toast.unsupportedFileType', + defaultMessage: 'You can only upload images and videos. This file type is not supported.', + description: 'Feedback when user tries to upload an unsupported file type', + }, + noRelays: { + id: 'toast.noRelays', + defaultMessage: 'You need to declare at least one relay to perform this action', + description: 'Toast message indicating user has no relays configured', + }, + noRelaysConnected: { + id: 'toast.noRelaysConnected', + defaultMessage: '"We are trying to connect to your relays. Please try again in a few moments.', + description: 'Toast message indicating user is not connected to aany relay', + }, + noteNostrLinkCoppied: { + id: 'noteNostrLinkCoppied', + defaultMessage: 'Note\'s nostr link copied', + description: 'Confirmation message that the note\'s link has been copied', + }, + notePrimalLinkCoppied: { + id: 'notePrimalLinkCoppied', + defaultMessage: 'Note\'s Primal link copied', + description: 'Confirmation message that the note\'s link has been copied', + }, + repostSuccess: { + id: 'toast.repostSuccess', + defaultMessage: 'Reposted successfully', + description: 'Toast message indicating successfull repost', + }, + repostFailed: { + id: 'toast.repostFailed', + defaultMessage: 'Failed to repost', + description: 'Toast message indicating failed repost', + }, + zapAsGuest: { + id: 'toast.zapAsGuest', + defaultMessage: 'You must be logged-in to perform a zap', + description: 'Toast message indicating user must be logged-in to perform a zap', + }, + zapUnavailable: { + id: 'toast.zapUnavailable', + defaultMessage: 'Author of this post cannot be zapped', + description: 'Toast message indicating user cannot receieve a zap', + }, +}; + +export const zapCustomOption = { + id: 'zap.custom.option', + defaultMessage: `Zap {user} `, + description: 'Caption for custom zap amount modal', +}; diff --git a/src/types/primal.d.ts b/src/types/primal.d.ts new file mode 100644 index 0000000..ef30b17 --- /dev/null +++ b/src/types/primal.d.ts @@ -0,0 +1,516 @@ +import { MessageDescriptor } from "@cookbook/solid-intl"; +import { Relay } from "nostr-tools"; +import { JSX } from "solid-js"; +import { SetStoreFunction } from "solid-js/store"; +import { Kind } from "../constants"; + + +export type NostrNoteContent = { + kind: Kind.Text | Kind.Repost, + content: string, + id: string, + created_at?: number, + pubkey: string, + sig: string, + tags: string[][], +}; + +export type NostrUserContent = { + kind: Kind.Metadata, + content: string, + id: string, + created_at?: number, + pubkey: string, + sig: string, + tags: string[][], +}; + +export type NostrStatsContent = { + kind: Kind.NoteStats, + content: string, + pubkey?: string, + created_at?: number, +}; + +export type NostrNetStatsContent = { + kind: Kind.NetStats, + content: string, + pubkey?: string, + created_at?: number, +}; + +export type NostrLegendStatsContent = { + kind: Kind.LegendStats, + content: string, + pubkey?: string, + created_at?: number, +}; + +export type NostrUserStatsContent = { + kind: Kind.UserStats, + content: string, + pubkey: string, + created_at: number, +}; + +export type NostrMentionContent = { + kind: Kind.Mentions, + content: string, + pubkey?: string, + created_at?: number, +}; + +export type NostrOldestEventContent = { + kind: Kind.OldestEvent, + content: string, + pubkey?: string, + created_at?: number, +}; + +export type NostrContactsContent = { + kind: Kind.Contacts, + content: string, + pubkey?: string, + created_at?: number, + tags: string[][], +}; + +export type NostrScoredUsersContent = { + kind: Kind.UserScore, + content: string, + created_at?: number, + pubkey?: string, +}; + +export type NostrNotificationContent = { + kind: Kind.Notification, + content: string, + created_at?: number, + pubkey?: string, +}; + +export type NostrNotificationLastSeenContent = { + kind: Kind.Timestamp, + content: string, + created_at?: number, + pubkey?: string, +}; + +export type NostrNotificationStatsContent = { + kind: Kind.NotificationStats, + content: string, + created_at?: number, + pubkey?: string, +}; + +export type NostrNoteActionsContent = { + kind: Kind.NoteActions, + content: string, + created_at?: number, + pubkey?: string, +}; + +export type NostrMessageStatsContent = { + kind: Kind.MessageStats, + cnt: string, + content?: string, + created_at?: number, + pubkey?: string, +}; + +export type NostrMessagePerSenderStatsContent = { + kind: Kind.MesagePerSenderStats, + content: string, + created_at?: number, + pubkey?: string, +}; + +export type NostrMessageEncryptedContent = { + kind: Kind.EncryptedDirectMessage, + content: string, + created_at: number, + pubkey: string, + id: string, +}; + +export type NostrFeedRange = { + kind: Kind.FeedRange, + content: string, +}; + +export type NostrMediaInfo = { + kind: Kind.MediaInfo, + content: string, +}; + +export type NostrMediaUploaded = { + kind: Kind.Uploaded, + content: string, +}; + +export type NostrEventContent = + NostrNoteContent | + NostrUserContent | + NostrStatsContent | + NostrNetStatsContent | + NostrLegendStatsContent | + NostrUserStatsContent | + NostrMentionContent | + NostrOldestEventContent | + NostrContactsContent | + NostrScoredUsersContent | + NostrNotificationContent | + NostrNotificationLastSeenContent | + NostrNotificationStatsContent | + NostrNoteActionsContent | + NostrMessageStatsContent | + NostrMessagePerSenderStatsContent | + NostrMessageEncryptedContent | + NostrFeedRange | + NostrMediaInfo | + NostrMediaUploaded; + +export type NostrEvent = [ + type: "EVENT", + subkey: string, + content: NostrEventContent, +]; + +export type NostrEOSE = [ + type: "EOSE", + subkey: string, +]; + +export type NoteActions = { + event_id: string, + liked: boolean, + replied: boolean, + reposted: boolean, + zapped: boolean, +}; + +export type FeedStore = { + posts: PrimalNote[], + isFetching: boolean, + scrollTop: number, + activeUser: PrimalUser | undefined, + publicKey: string | undefined, + selectedFeed: PrimalFeed | undefined, + availableFeeds: PrimalFeed[], + showNewNoteForm: boolean, + theme: string, + trendingNotes: TrendingNotesStore, + zappedNotes: TrendingNotesStore, + exploredNotes: PrimalNote[] | [], + threadedNotes: PrimalNote[] | [], +}; + +export type NostrPostStats = { + [eventId: string]: { + likes: number, + mentions: number, + reposts: number, + replies: number, + zaps: number, + satszapped: number, + score: number, + score24h: number, + }, +}; + +export type FeedPage = { + users: { + [pubkey: string]: NostrUserContent, + }, + messages: NostrNoteContent[], + postStats: NostrPostStats, + mentions?: Record, + noteActions: Record, +}; + +export type TrendingNotesStore = { + users: { + [pubkey: string]: NostrUserContent, + }, + messages: NostrNoteContent[], + notes: PrimalNote[], + postStats: NostrPostStats, +}; + +export type PrimalContextStore = { + + data: FeedStore, + page: FeedPage, + relays: Relay[], + // likes: string[], + actions?: { + clearThreadedNotes: () => void, + setThreadedNotes: (newNotes: PrimalNote[]) => void, + setData: SetStoreFunction, + clearExploredNotes: () => void, + setExploredNotes: (newNotes: PrimalNote[]) => void, + clearTrendingNotes: () => void, + clearZappedNotes: () => void, + setTheme: (newTheme: string) => void, + showNewNoteForm: () => void, + hideNewNoteForm: () => void, + fetchHomeFeed: () => void, + selectFeed: (profile: PrimalFeed | undefined) => void, + clearData: () => void, + loadNextPage: () => void, + savePosts: (posts: PrimalNote[]) => void, + clearPage: () => void, + setActiveUser: (user: PrimalUser) => void, + updatedFeedScroll: (scrollTop: number) => void, + proccessEventContent: ( + content: NostrUserContent | NostrNoteContent | NostrStatsContent, + type: string + ) => void, + }, +}; + +export type NostrRelay = { read: boolean, write: boolean }; + +export type NostrRelays = Record; + +export type NostrRelayEvent = { + kind: number, + content: string, + created_at: number, + tags: string[][], +}; +export type NostrRelaySignedEvent = NostrRelayEvent & { + id: string, + pubkey: string, + sig: string, +}; + +interface SendPaymentResponse { + preimage: string; +} + +export type NostrWindow = Window & typeof globalThis & { + nostr?: { + getPublicKey: () => Promise, + getRelays: () => Promise, + signEvent: (event: NostrRelayEvent) => Promise, + nip04: { + encrypt: (pubkey: string, message: string) => Promise, + decrypt: (pubkey: string, message: string) => Promise, + }, + }, + webln?: { + enable: () => Promise, + sendPayment: (req: string) => Promise; + }, +}; + +export type PrimalWindow = Window & typeof globalThis & { + loadPrimalStores: () => void, + primal?: any, +}; + +export type NostrEventType = "EVENT" | "EOSE" | "NOTICE"; + +export type NostrMessage = [ + type: NostrEventType, + subkey: string, + info: { + kind: number, + content: string, + }, +]; + +export type PrimalUser = { + id: string, + pubkey: string, + npub: string, + name: string, + about: string, + picture: string, + nip05: string, + banner: string, + display_name?: string, + displayName: string, + location: string, + lud06: string, + lud16: string, + website: string, + tags: string[][], +}; + +export type PrimalNoteData = { + id: string, + pubkey: string, + created_at: number, + tags: string[][], + content: string, + sig: string, + likes: number, + mentions: number, + reposts: number, + replies: number, + zaps: number, + score: number, + score24h: number, + satszapped: number, + noteId: string, + noteActions: NoteActions, +} + +export type PrimalNote = { + user: PrimalUser, + post: PrimalNoteData, + repost?: PrimalRepost, + msg: NostrNoteContent, + mentionedNotes?: Record, + mentionedUsers?: Record, +}; + +export type PrimalFeed = { + name: string, + npub?: string, + hex?: string, +}; + +export type PrimalScopeFeed = { + name: string, + scope?: string, + timeframe?: string, +}; + +// export type PrimalFeed = PrimalUserFeed & PrimalScopeFeed; + +export type PrimalNetStats = { + users: number, + pubkeys: number, + pubnotes: number, + reactions: number, + reposts: number, + any: number, + zaps: number, + satszapped: number, +}; + +export type PrimalResponse = { + op: string, + netstats?: PrimalNetStats; +}; + +export type PrimalLegend = { + your_follows: number, + your_inner_network: number, + your_outer_network: number, +}; + +export type FeedOption = { + label: string, + value: string | undefined, +}; + +export type PrimalRepost = { + user: PrimalUser, + note: PrimalNoteData, +} + +export type RepostInfo = (page: FeedPage, message: NostrNoteContent) => PrimalRepost; + +export type ExploreFeedPayload = { + timeframe: string, + scope: string, + limit: number, + user_pubkey?: string, + since? : number, + until?: number, + created_after?: number, +} + +export type UserReference = { + id: string, + pubkey: string, + kind: number, + tags: string[][], + npub?: string, + name?: string, + about?: string, + picture?: string, + nip05?: string, + banner?: string, + display_name?: string, + displayName?: string, + location?: string, + lud06?: string, + lud16?: string, + website?: string, + content?: string, + created_at?: number, + sig?: string, +}; + +export type ContextChildren = + number | + boolean | + Node | + JSX.ArrayElement | + JSX.FunctionElement | + (string & {}) | null | undefined; + + +export type PrimalTheme = { name: string, label: string, logo: string, dark?: boolean}; + +export type ChildrenProp = { children: number | boolean | Node | JSX.ArrayElement | JSX.FunctionElement | (string & {}) | null | undefined; }; + +export type VanityProfiles = { names: Record }; + +export type PrimalNotifUser = PrimalUser & { followers_count: number }; + +export type PrimalNotification = { + pubkey: string, + created_at: number, + type: number, + your_post?: string, + follower?: string, + you_were_mentioned_in?: string, + your_post_were_mentioned_in?: string, + post_you_were_mentioned_in?: string, + post_your_post_was_mentioned_in?: string, + who_liked_it?: string, + who_zapped_it?: string, + who_reposted_it?: string, + who_replied_to_it?: string, + satszapped?: number, +}; + +export type SortedNotifications = Record; + +export type UserRelation = 'follows' | 'other' | 'any'; + +export type EmojiOption = { + keywords: string[], + char: string, + fitzpatrick_scale: boolean, + category: string, + name: string, +}; + +export type MediaSize = 'o' | 's' | 'm' | 'l'; + +export type MediaVariant = { + s: MediaSize, + a: 0 | 1, + w: number, + h: number, + mt: string, + media_url: string, +} + +export type MediaEvent = { + event_id: string, + resources: { url: string, variants: MediaVariant[] }[], +} + +export type ScopeDescriptor = { + caption: MessageDescriptor, + label: MessageDescriptor, + description: MessageDescriptor, +} diff --git a/src/uploadSocket.tsx b/src/uploadSocket.tsx new file mode 100644 index 0000000..bb16f54 --- /dev/null +++ b/src/uploadSocket.tsx @@ -0,0 +1,102 @@ +import { createSignal } from "solid-js"; +import { NostrEvent, NostrEOSE, NostrEventType, NostrEventContent } from "./types/primal"; + +export const [socket, setSocket] = createSignal(); + +export const [isConnected, setConnected] = createSignal(false); + +export const isNotConnected = () => !isConnected(); + +const onOpen = () => { + setConnected(true); +} + +const onClose = () => { + setConnected(false); + + socket()?.removeEventListener('open', onOpen); + socket()?.removeEventListener('close', onClose); + socket()?.removeEventListener('error', onError); + + setTimeout(() => { + connect(); + }, 200); +} + +const onError = (error: Event) => { + console.log("ws error: ", error); +}; + +export const connect = () => { + if (isNotConnected()) { + const cacheServer = localStorage.getItem('uploadServer') ?? + 'wss://uploads.primal.net/v1'; + + setSocket(new WebSocket(cacheServer)); + console.log('UPLOAD SOCKET: ', socket()); + + socket()?.addEventListener('open', onOpen); + socket()?.addEventListener('close', onClose); + socket()?.addEventListener('error', onError); + } +}; + +export const disconnect = () => { + socket()?.close(); +}; + +export const reset = () => { + disconnect(); + setTimeout(connect, 1000); +}; + +export const sendMessage = (message: string) => { + isConnected() && socket()?.send(message); +} + +export const refreshSocketListeners = ( + ws: WebSocket | undefined, + listeners: Record any>, + ) => { + + if (!ws) { + return; + } + + Object.keys(listeners).forEach((event: string) => { + ws.removeEventListener(event, listeners[event]); + ws.addEventListener(event, listeners[event]); + }); +}; + +export const removeSocketListeners = ( + ws: WebSocket | undefined, + listeners: Record any>, + ) => { + + if (!ws) { + return; + } + + Object.keys(listeners).forEach((event: string) => { + ws.removeEventListener(event, listeners[event]); + }); +}; + +export const subscribeTo = (subId: string, cb: (type: NostrEventType, subId: string, content?: NostrEventContent) => void ) => { + const listener = (event: MessageEvent) => { + const message: NostrEvent | NostrEOSE = JSON.parse(event.data); + const [type, subscriptionId, content] = message; + + if (subId === subscriptionId) { + cb(type, subscriptionId, content); + } + + }; + + socket()?.addEventListener('message', listener); + + return () => { + socket()?.removeEventListener('message', listener); + }; +}; diff --git a/src/utils.ts b/src/utils.ts new file mode 100644 index 0000000..803366b --- /dev/null +++ b/src/utils.ts @@ -0,0 +1,23 @@ +let debounceTimer: number = 0; + +export const debounce = (callback: TimerHandler, time: number) => { + if (debounceTimer) { + window.clearTimeout(debounceTimer); + } + + debounceTimer = window.setTimeout(callback, time); +} + +export const isVisibleInContainer = (element: Element, container: Element) => { + const { bottom, height, top } = element.getBoundingClientRect(); + const containerRect = container.getBoundingClientRect(); + + return top <= containerRect.top ? containerRect.top - top <= height : bottom - containerRect.bottom <= height; +}; + +export const uuidv4 = () => { + // @ts-ignore + return ([1e7]+-1e3+-4e3+-8e3+-1e11).replace(/[018]/g, c => + (c ^ crypto.getRandomValues(new Uint8Array(1))[0] & 15 >> c / 4).toString(16) + ); +} diff --git a/tsconfig.json b/tsconfig.json new file mode 100644 index 0000000..30ac248 --- /dev/null +++ b/tsconfig.json @@ -0,0 +1,16 @@ +{ + "compilerOptions": { + "strict": true, + "target": "ESNext", + "module": "ESNext", + "moduleResolution": "node", + "allowSyntheticDefaultImports": true, + "esModuleInterop": true, + "jsx": "preserve", + "jsxImportSource": "solid-js", + "types": ["vite/client"], + "noEmit": true, + "isolatedModules": true, + "resolveJsonModule": true + } +} diff --git a/vite.config.ts b/vite.config.ts new file mode 100644 index 0000000..476f5ba --- /dev/null +++ b/vite.config.ts @@ -0,0 +1,13 @@ +import { defineConfig } from 'vite'; +import solidPlugin from 'vite-plugin-solid'; + +export default defineConfig({ + plugins: [solidPlugin()], + server: { + port: 3000, + }, + build: { + target: 'esnext', + sourcemap: true, + }, +});