diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml new file mode 100644 index 0000000000000000000000000000000000000000..d7faa7714218e08c70603e9285164946845b8369 --- /dev/null +++ b/.gitlab-ci.yml @@ -0,0 +1,42 @@ +stages: + - build_staging + - restart_staging + - clean_up + +build_staging: + stage: build_staging + script: + - echo "Building current develop branch" + - ./run.sh build stg + environment: + name: staging + only: + - develop + +restart_staging: + stage: restart_staging + script: + - echo "Stopping current condenser image" + - ./run.sh stop stg + - echo "Starting latest condenser image" + - ./run.sh start stg + dependencies: + - build_staging + environment: + name: staging + only: + - develop + +clean_up: + stage: clean_up + script: + - echo "Cleanup filespace / unused images / containers / networks" + - docker image prune -f + - docker container prune -f + - docker network prune -f + dependencies: + - restart_staging + environment: + name: staging + only: + - develop diff --git a/README.md b/README.md index 4004b2fa706805257a1aebadf2313ef2b063a245..4d1c3638a798615475289048957311ad0442e829 100644 --- a/README.md +++ b/README.md @@ -1,17 +1,16 @@ # Condenser - -Condenser is the react.js web interface to the world's first and used to be best -blockchain-based social media platform, steemit.com. It uses -[Steen compatible blockchain](https://github.com/steemit/steem), powered by DPoS Governance and ChainBase DB to store JSON-based content for a plethora of web +Condenser is the react.js web interface to the +blockchain-based social media platform, Hive.blog. It uses a +[Hive compatible blockchain](https://gitlab.syncad.com/hive/hive), powered by DPoS Governance and ChainBase DB to store JSON-based content for a plethora of web applications. ## Why would I want to use Condenser? -* Learning how to build blockchain-based web applications using Steem compatible blockchain as a content storage mechanism in react.js -* Reviewing the inner workings of the steemit.com social media platform -* Assisting with software development for steemit.com +* Learning how to build blockchain-based web applications using Hive compatible blockchain as a content storage mechanism in react.js +* Reviewing the inner workings of the Hive.blog social media platform +* Assisting with software development for Hive.blog and Hive.io ## Installation @@ -22,19 +21,31 @@ simple as pulling in the github repo and issuing one command to build it, like this: ```bash -git clone https://github.com/steemit/condenser +git clone https://gitlab.syncad.com/hive/condenser cd condenser docker build -t="myname/condenser:mybranch" . docker run -it -p 8080:8080 myname/condenser:mybranch ``` +#### Docker Compose + +If you like to run and build condenser and additionally a reverse-proxy using an Nginx Docker image, with companion Letsencrypt (SSL) support, you can simple launch the Docker-compose files via the included `run.sh`-scripts. + +```bash +git clone https://gitlab.syncad.com/hive/condenser +cd condenser +./run.sh start proxy # to start the nginx reverse proxy (with ssl support) +./run.sh start (prod|dev|stg) # to build and start the condensor image +./run.sh logs (prod|dev|stg) # (optionally) to attach to the condensor image and inspect its logs +``` + ## Building from source without docker (the 'traditional' way): (better if you're planning to do condenser development) #### Clone the repository and make a tmp folder ```bash -git clone https://github.com/steemit/condenser +git clone https://gitlab.syncad.com/hive/condenser cd condenser mkdir tmp ``` @@ -47,8 +58,8 @@ Condenser is known to successfully build using node 12.6, npm 6.13.4, and yarn 1.22.4. We use the yarn package manager instead of the default `npm`. There are -multiple reasons for this, one being that we have `steem-js` built from -source pulling the github repo as part of the build process and yarn +multiple reasons for this, one being that we have `hive-js` built from +source pulling the gitlab repo as part of the build process and yarn supports this. This way the library that handles keys can be loaded by commit hash instead of a version name and cryptographically verified to be exactly what we expect it to be. Yarn can be installed with `npm`, but @@ -80,12 +91,12 @@ It will take quite a bit longer to start in this mode (~60s) as it needs to build and start the webpack-dev-server. By default you will be connected to community public api node at -`https://api.steem.house`. This is actually on the real blockchain and +`https://api.hive.blog`. This is actually on the real blockchain and you would use your regular account name and credentials to login - there is not an official separate testnet at this time. If you intend to run a full-fledged site relying on your own, we recommend looking into running a -copy of `steemd` locally instead -[https://github.com/steemit/steem](https://github.com/steemit/steem). +copy of `hive (steemd)` locally instead +[https://gitlab.syncad.com/hive/hive](https://gitlab.syncad.com/hive/hive). #### Debugging SSR code @@ -104,8 +115,8 @@ stored in `config/defaults.json`. Environment variables using an example like this: ```bash -export SDC_CLIENT_STEEMD_URL="https://api.steem.house" -export SDC_SERVER_STEEMD_URL="https://api.steem.house" +export SDC_CLIENT_STEEMD_URL="https://api.hive.blog" +export SDC_SERVER_STEEMD_URL="https://api.hive.blog" ``` Keep in mind environment variables only exist in your active session, so if @@ -187,16 +198,6 @@ OFFLINE_SSR_TEST=true NODE_ENV=production node --prof lib/server/index.js This will read data from the blobs in `api_mockdata` directory. If you want to use another set of mock data, create a similar directory to that one and add an argument `OFFLINE_SSR_TEST_DATA_DIR` pointing to your new directory. -### Run blackbox tests using nightwatch - -To run a Selenium test suite, start the condenser docker image with a name `condenser` (like `docker run --name condenser -itp 8080:8080 steemit/condenser:latest`) and then run the blackboxtest image attached to the condneser image's network: - -``` -docker build -t=steemit/condenser-blackboxtest blackboxtest/ -docker run --network container:condenser steemit/condenser-blackboxtest:latest - -``` - ## Issues To report a non-critical issue, please file an issue on this GitHub project. diff --git a/config/default.json b/config/default.json index a6c916ebaa968f732798d5e331f6424127868f47..1cfbfdcb2e565fa485f62891d8f1aed7aec063e9 100644 --- a/config/default.json +++ b/config/default.json @@ -4,7 +4,7 @@ "helmet": { "directives": { "childSrc": "'self' 3speak.online emb.d.tube player.twitch.tv www.youtube.com staticxx.facebook.com w.soundcloud.com player.vimeo.com", - "connectSrc": "https://images.hive.blog 'self' hive.blog https://api.hive.blog api.blocktrades.us", + "connectSrc": "https://images.hive.blog 'self' hive.blog https://api.hive.blog api.blocktrades.us https://anyx.io", "defaultSrc": "tpc.googlesyndication.com 'self' img.3speakcontent.online emb.d.tube www.youtube.com staticxx.facebook.com player.vimeo.com *.streamrail.com", "fontSrc": "data: fonts.gstatic.com", "frameAncestors": "'none'", @@ -39,6 +39,8 @@ "steemd_connection_server": "https://api.hive.blog", "steemd_use_appbase": false, "chain_id": "0000000000000000000000000000000000000000000000000000000000000000", + "alternative_api_endpoints": ["https://api.hive.blog", "https://anyx.io"], + "failover_threshold": 3, "address_prefix": "STM", "conveyor_posting_wif": false, "conveyor_username": false, diff --git a/docker-compose.dev.yml b/docker-compose.dev.yml index d576d9996323f910fc2aae99b2f4782c6f0d416b..4beefcfd21cf5beac471cb5c986906f768f7aea6 100644 --- a/docker-compose.dev.yml +++ b/docker-compose.dev.yml @@ -19,7 +19,7 @@ services: SDC_IMAGE_PROXY_PREFIX: https://images.hive.blog/ SDC_UPLOAD_IMAGE_URL: https://images.hive.blog - SDC_HELMET_CONNECTSRC: "'self' http://hiveblog.local https://api.hive.blog https://images.hive.blog" + SDC_HELMET_CONNECTSRC: "'self' http://hiveblog.local https://api.hive.blog https://images.hive.blog https://anyx.io" volumes: - ./yarn.lock:/var/app/yarn.lock - ./package.json:/var/app/package.json diff --git a/docker-compose.proxy.yml b/docker-compose.proxy.yml index a3e44df14ccb589df39748c3588ce28d9b6bd8d7..bd14cdafc7149214c680e0f80600afcbfc190ffe 100644 --- a/docker-compose.proxy.yml +++ b/docker-compose.proxy.yml @@ -3,16 +3,60 @@ services: proxy: image: jwilder/nginx-proxy:latest container_name: proxy + labels: + com.github.jrcs.letsencrypt_nginx_proxy_companion.nginx_proxy: "true" ports: - "80:80" - "443:443" volumes: + - letsencrypt-certs:/etc/nginx/certs:ro + - nginx-vhost:/etc/nginx/vhost.d + - nginx-html:/usr/share/nginx/html + - nginx-conf:/etc/nginx/conf.d - /var/run/docker.sock:/tmp/docker.sock:ro - ./client_max_body_size.conf:/etc/nginx/conf.d/client_max_body_size.conf:ro networks: - reverse-proxy restart: always + letsencrypt: + image: jrcs/letsencrypt-nginx-proxy-companion:latest + container_name: letsencrypt + depends_on: + - proxy + environment: + NGINX_PROXY_CONTAINER: proxy + volumes: + - letsencrypt-certs:/etc/nginx/certs + - nginx-vhost:/etc/nginx/vhost.d + - nginx-html:/usr/share/nginx/html + - /var/run/docker.sock:/var/run/docker.sock:ro + networks: + - reverse-proxy + restart: always + networks: reverse-proxy: driver: bridge + +volumes: + letsencrypt-certs: + driver_opts: + type: none + device: /etc/letsencrypt/certs + o: bind + nginx-conf: + driver_opts: + type: none + device: /etc/nginx/conf.d + o: bind + nginx-vhost: + driver_opts: + type: none + device: /etc/nginx/vhost.d + o: bind + nginx-html: + driver_opts: + type: none + device: /var/www/html + o: bind diff --git a/docker-compose.staging.yml b/docker-compose.staging.yml new file mode 100644 index 0000000000000000000000000000000000000000..9a0f0e082d582d65cec3d7e9a4811c45f8fde430 --- /dev/null +++ b/docker-compose.staging.yml @@ -0,0 +1,27 @@ +version: "3" +services: + hive_condenser_stg: + build: . + image: hive/condenser:staging + container_name: hive_condenser_stg + environment: + VIRTUAL_HOST: staging-blog.hive.io + VIRTUAL_PORT: 8080 + LETSENCRYPT_HOST: staging-blog.hive.io + LETSENCRYPT_EMAIL: certs@hive.io + SDC_CLIENT_STEEMD_URL: https://api.hive.blog + SDC_SERVER_STEEMD_URL: https://api.hive.blog + SDC_DISABLE_SIGNUPS: 1 + SDC_SITE_DOMAIN: staging-blog.hive.io + SDC_IMAGE_PROXY_PREFIX: https://images.hive.blog/ + SDC_UPLOAD_IMAGE_URL: https://images.hive.blog + SDC_HELMET_CONNECTSRC: "'self' https://api.hive.blog https://staging.hive.blog https://images.hive.blog https://anyx.io" + WALLET_URL: https://wallet.hive.blog + networks: + - condenser_reverse-proxy + restart: always + +networks: + condenser_reverse-proxy: + external: + name: condenser_reverse-proxy diff --git a/package.json b/package.json index b2081b44a006bea03c093607e4228729f1538cb5..69e7c43ad27f25fb0944a88c92dbfb2508d08b44 100644 --- a/package.json +++ b/package.json @@ -24,7 +24,7 @@ "license": "MIT", "dependencies": { "@hivechain/hivescript": "^1.1.3", - "@hiveio/hive-js": "0.0.1", + "@hiveio/hive-js": "^0.0.2", "@steem/crypto-session": "git+https://github.com/steemit/crypto-session.git#83a90b319ce5bc6a70362d52a15a815de7e729bb", "assert": "1.4.1", "autoprefixer-loader": "3.2.0", diff --git a/run.sh b/run.sh index 47fe5dc34f2f32fa419e98fcf3711236e9fe59c0..341368ee4b8ecd1666fadc697983531788196a3e 100755 --- a/run.sh +++ b/run.sh @@ -10,6 +10,10 @@ function start { docker-compose -f docker-compose.prod.yml up -d ;; + "stg") + docker-compose -f docker-compose.staging.yml up -d + ;; + "dev") docker-compose -f docker-compose.dev.yml up ;; @@ -31,6 +35,10 @@ function stop { docker-compose -f docker-compose.prod.yml down ;; + "stg") + docker-compose -f docker-compose.staging.yml down + ;; + "dev") docker-compose -f docker-compose.dev.yml down ;; @@ -52,6 +60,10 @@ function logs { docker-compose -f docker-compose.prod.yml logs --tail 30 --follow ;; + "stg") + docker-compose -f docker-compose.staging.yml logs --tail 30 --follow + ;; + "*") echo Unknown environment exit 1 @@ -70,6 +82,10 @@ function build { docker-compose -f docker-compose.prod.yml build ;; + "stg") + docker-compose -f docker-compose.staging.yml build + ;; + "*") echo Unknown environment exit 1 @@ -124,7 +140,7 @@ while test $# -gt 0; do ;; *) - echo "Usage: ./run.sh <start|stop|log> <prod|dev>" + echo "Usage: ./run.sh <start|stop|log> <prod|stg|dev>" exit 1 ;; esac diff --git a/src/app/Main.js b/src/app/Main.js index bacfa23879f540f62c25e2dbe0fd53ec9636da39..5d632dbad9185fd2ff8321236180403c1a7fdbaf 100644 --- a/src/app/Main.js +++ b/src/app/Main.js @@ -85,12 +85,22 @@ function runApp(initial_state) { const config = initial_state.offchain.config; steem.api.setOptions({ - url: config.steemd_connection_client, + url: + localStorage.getItem('user_preferred_api_endpoint') === null + ? config.steemd_connection_client + : localStorage.getItem('user_preferred_api_endpoint'), retry: true, useAppbaseApi: !!config.steemd_use_appbase, + alternative_api_endpoints: config.alternative_api_endpoints, + failover_threshold: config.failover_threshold, }); steem.config.set('address_prefix', config.address_prefix); steem.config.set('chain_id', config.chain_id); + steem.config.set('failover_threshold', config.failover_threshold); + steem.config.set( + 'alternative_api_endpoints', + config.alternative_api_endpoints + ); window.$STM_Config = config; plugins(config); if (initial_state.offchain.serverBusy) { diff --git a/src/app/assets/images/hive-blog-logo-nightmode.svg b/src/app/assets/images/hive-blog-logo-nightmode.svg index fb0675bfceedfaddde5b2fcae8e45c4ab48ef35c..90e099c4908d54377e7af14df6894a8e14904a0d 100644 --- a/src/app/assets/images/hive-blog-logo-nightmode.svg +++ b/src/app/assets/images/hive-blog-logo-nightmode.svg @@ -1,15 +1,10 @@ -<?xml version="1.0" encoding="UTF-8" standalone="no"?> -<!-- Generator: GIMP export as svg plugin --> - -<svg xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:cc="http://creativecommons.org/ns#" xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" xmlns:svg="http://www.w3.org/2000/svg" xmlns="http://www.w3.org/2000/svg" xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" xmlns:xlink="http://www.w3.org/1999/xlink" xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" width="373px" height="109px" viewBox="0 0 373 109" version="1.1"><defs id="defs2" /> -<sodipodi:namedview id="base" pagecolor="#ffffff" bordercolor="#666666" borderopacity="1.0" inkscape:pageopacity="0.0" inkscape:pageshadow="2" inkscape:document-units="px" showgrid="false" inkscape:window-maximized="1" /> -<metadata id="metadata5"> -<rdf:RDF> -<cc:Work rdf:about=""> -<dc:format>image/svg+xml</dc:format> -<dc:type rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> -<dc:title></dc:title> -</cc:Work> -</rdf:RDF> -</metadata><g inkscape:groupmode="layer" inkscape:label="hive-blog-logo.png" ><image xlink:href="" x="0" y="0" width="373" height="109" /> -</g></svg> \ No newline at end of file +<svg xmlns="http://www.w3.org/2000/svg" width="373" height="109" viewBox="0 0 373 109"> + <g fill="none"> + <g fill="#E31337"> + <path d="M90.4422108,61.5356296 C90.85992,61.5356296 91.1207865,61.9874183 90.9115288,62.3484305 L64.0269732,108.729763 C63.930027,108.897015 63.7511741,109 63.5576553,109 L47.1234647,109 C46.7057556,109 46.4448891,108.548211 46.6541468,108.187199 L73.5387023,61.8058664 C73.6356485,61.6386145 73.8145015,61.5356296 74.0080203,61.5356296 L90.4422108,61.5356296 Z M74.4582937,48.2411837 C74.2647749,48.2411837 74.0859219,48.1381988 73.9889757,47.9709469 L46.6541468,0.812800858 C46.4448891,0.451788656 46.7057556,0 47.1234647,0 L63.5576553,0 C63.7511741,0 63.930027,0.102984848 64.0269732,0.270236757 L91.3618022,47.4283828 C91.5710599,47.789395 91.3101934,48.2411837 90.8924842,48.2411837 L74.4582937,48.2411837 Z"/> + <path d="M77.7078135.812538775C77.4981696.45152632 77.7598718 0 78.1787565 0L94.6363581 0C94.8305929 0 95.0100903.10309919 95.1073011.27049884L126.44159 54.22898C126.538963 54.3966589 126.538963 54.6033411 126.44159 54.77102L95.1073011 108.729501C95.0100903 108.896901 94.8305929 109 94.6363581 109L78.1787565 109C77.7598718 109 77.4981696 108.548474 77.7078135 108.187461L108.884718 54.5 77.7078135.812538775zM64.3328086 54.2269735C64.431852 54.3948735 64.4323504 54.6024365 64.3341146 54.7708006L32.850801 108.729282C32.6407695 109.089249 32.1162201 109.090483 31.9044622 108.731508L.0746342025 54.7730265C-.0244091387 54.6051265-.024907567 54.3975635.0733282916 54.2291994L31.5566419.270718284C31.7666733-.0892494466 32.2912227-.0904832685 32.5029806.268492348L64.3328086 54.2269735z"/> + </g> + <path fill="#FFFFFF" d="M186.740404,0 L199.453228,0 L199.453228,46.87 L186.740404,46.87 L186.740404,28.5907 L168.674812,28.5907 L168.674812,46.87 L155.961988,46.87 L155.961988,0 L168.674812,0 L168.674812,16.9401571 L186.740404,16.9401571 L186.740404,0 Z M226.217067,0 L238.762617,0 L238.762617,46.87 L226.217067,46.87 L226.217067,0 Z M279.750831,46.87 L261.344607,0.602614286 L261.344607,0 L275.132542,0 L286.644799,31.8046429 L298.157056,0 L311.944991,0 L311.944991,0.602614286 L293.538767,46.87 L279.750831,46.87 Z M346.849018,35.3533714 L373,35.3533714 L373,46.87 L334.108796,46.87 L334.108796,0 L372.530623,0 L372.530623,11.5166286 L346.849018,11.5166286 L346.849018,18.0114714 L363.143091,18.0114714 L363.143091,28.7246143 L346.849018,28.7246143 L346.849018,35.3533714 Z"/> + <path fill="#E31337" d="M175.27868,81.8564453 C178.243977,82.9634821 180.478506,84.538857 181.982335,86.5826172 C183.486164,88.6263774 184.238067,91.0533063 184.238067,93.8634766 C184.238067,97.7806836 182.829573,101.027233 180.012541,103.603223 C177.195509,106.179212 173.520715,107.467187 168.988047,107.467187 L155.961988,107.467187 L155.961988,60.7164062 L165.429709,60.7164062 C170.343629,60.7164062 174.081965,61.7276266 176.644828,63.7500977 C179.207692,65.7725687 180.489104,68.6572078 180.489104,72.4041016 C180.489104,76.491622 178.752314,79.6423717 175.27868,81.8564453 Z M161.311517,79.8765625 L163.507299,79.8765625 C167.382228,79.8765625 170.277618,79.2804747 172.193556,78.0882812 C174.109493,76.8960878 175.067447,75.0013802 175.067447,72.4041016 C175.067447,68.1888461 172.11824,66.08125 166.219736,66.08125 L161.311517,66.08125 L161.311517,79.8765625 Z M161.311517,102.102344 L166.630025,102.102344 C169.959515,102.102344 172.380925,101.780573 173.89433,101.137023 C175.407734,100.493472 176.618439,99.5067099 177.526482,98.1767056 C178.434524,96.8467014 178.888539,95.4952657 178.888539,94.1223581 C178.888539,92.7494504 178.618292,91.5374486 178.077791,90.4863162 C177.537289,89.4351838 176.737359,88.5020496 175.677976,87.6868857 C174.618593,86.8717218 173.321409,86.2603581 171.786384,85.8527761 C170.25136,85.4451942 167.473224,85.2414062 163.451892,85.2414062 L161.311517,85.2414062 L161.311517,102.102344 Z M231.619605,107.467187 L207.928836,107.467187 L207.928836,60.7164062 L213.606789,60.7164062 L213.606789,102.166211 L231.619605,102.166211 L231.619605,107.467187 Z M276.423021,59.95 C280.948221,59.95 285.113879,61.0250869 288.920122,63.175293 C292.726365,65.325499 295.71845,68.2952936 297.896466,72.0847656 C300.074483,75.8742377 301.163475,79.9936301 301.163475,84.4430664 C301.163475,88.8925027 300.085056,93.0118951 297.928185,96.8013672 C295.771314,100.590839 292.810947,103.571278 289.046996,105.742773 C285.283045,107.914269 281.191396,109 276.771925,109 C272.352454,109 268.239659,107.924913 264.433416,105.774707 C260.627174,103.624501 257.645662,100.665351 255.488791,96.897168 C253.33192,93.1289851 252.253501,89.0202371 252.253501,84.5708008 C252.253501,80.1213645 253.33192,75.9913276 255.488791,72.1805664 C257.645662,68.3698052 260.595455,65.3787218 264.338261,63.2072266 C268.081066,61.0357313 272.109279,59.95 276.423021,59.95 Z M276.611833,103.635156 C281.981578,103.635156 286.524315,101.825604 290.240179,98.2064453 C293.956042,94.5872866 295.813947,90.0527616 295.813947,84.6027344 C295.813947,79.1527071 293.934564,74.5756045 290.175742,70.8712891 C286.41692,67.1669737 281.895662,65.3148437 276.611833,65.3148437 C271.199129,65.3148437 266.677872,67.1776181 263.047924,70.9032227 C259.417976,74.6288272 257.603029,79.1633522 257.603029,84.5069336 C257.603029,89.850515 259.471672,94.3743955 263.209015,98.0787109 C266.946358,101.783026 271.413919,103.635156 276.611833,103.635156 Z M373,84.9859375 C372.87263,92.5648816 370.675524,98.461893 366.408617,102.677148 C362.14171,106.892404 356.526884,109 349.563971,109 C341.582095,109 335.213672,106.541138 330.458512,101.62334 C325.703352,96.7055418 323.325807,90.946908 323.325807,84.3472656 C323.325807,77.7476233 325.703352,72.0315671 330.458512,67.1989258 C335.213672,62.3662844 341.348586,59.95 348.863437,59.95 C352.939289,59.95 356.7285,60.7057542 360.231185,62.2172852 C363.73387,63.7288162 367.162204,66.0386563 370.51629,69.146875 L366.313089,73.1705078 C363.723225,70.6583859 360.921119,68.731745 357.906688,67.3905273 C354.892256,66.0493097 351.792957,65.3787109 348.608698,65.3787109 C345.424439,65.3787109 342.261455,66.2302649 339.119653,67.9333984 C335.977851,69.636532 333.515394,71.9357277 331.732209,74.8310547 C329.949024,77.7263817 329.057445,80.8558426 329.057445,84.2195313 C329.057445,89.4566668 331.021042,94.0231251 334.948295,97.919043 C338.875547,101.814961 343.789847,103.762891 349.69134,103.762891 C354.064389,103.762891 357.842987,102.560071 361.027246,100.154395 C364.211505,97.7487184 366.164488,94.4595912 366.886253,90.2869141 L352.366105,90.2869141 L352.366105,84.9859375 L373,84.9859375 Z"/> + </g> +</svg> diff --git a/src/app/assets/images/hive-blog-logo.svg b/src/app/assets/images/hive-blog-logo.svg index fb0675bfceedfaddde5b2fcae8e45c4ab48ef35c..d4e160862519e20f412993fe8f2c8e7607a2fbb2 100644 --- a/src/app/assets/images/hive-blog-logo.svg +++ b/src/app/assets/images/hive-blog-logo.svg @@ -1,15 +1,10 @@ -<?xml version="1.0" encoding="UTF-8" standalone="no"?> -<!-- Generator: GIMP export as svg plugin --> - -<svg xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:cc="http://creativecommons.org/ns#" xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" xmlns:svg="http://www.w3.org/2000/svg" xmlns="http://www.w3.org/2000/svg" xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" xmlns:xlink="http://www.w3.org/1999/xlink" xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" width="373px" height="109px" viewBox="0 0 373 109" version="1.1"><defs id="defs2" /> -<sodipodi:namedview id="base" pagecolor="#ffffff" bordercolor="#666666" borderopacity="1.0" inkscape:pageopacity="0.0" inkscape:pageshadow="2" inkscape:document-units="px" showgrid="false" inkscape:window-maximized="1" /> -<metadata id="metadata5"> -<rdf:RDF> -<cc:Work rdf:about=""> -<dc:format>image/svg+xml</dc:format> -<dc:type rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> -<dc:title></dc:title> -</cc:Work> -</rdf:RDF> -</metadata><g inkscape:groupmode="layer" inkscape:label="hive-blog-logo.png" ><image xlink:href="" x="0" y="0" width="373" height="109" /> -</g></svg> \ No newline at end of file +<svg xmlns="http://www.w3.org/2000/svg" width="373" height="109" viewBox="0 0 373 109"> + <g fill="none"> + <g fill="#E31337"> + <path d="M90.4422108,61.5356296 C90.85992,61.5356296 91.1207865,61.9874183 90.9115288,62.3484305 L64.0269732,108.729763 C63.930027,108.897015 63.7511741,109 63.5576553,109 L47.1234647,109 C46.7057556,109 46.4448891,108.548211 46.6541468,108.187199 L73.5387023,61.8058664 C73.6356485,61.6386145 73.8145015,61.5356296 74.0080203,61.5356296 L90.4422108,61.5356296 Z M74.4582937,48.2411837 C74.2647749,48.2411837 74.0859219,48.1381988 73.9889757,47.9709469 L46.6541468,0.812800858 C46.4448891,0.451788656 46.7057556,0 47.1234647,0 L63.5576553,0 C63.7511741,0 63.930027,0.102984848 64.0269732,0.270236757 L91.3618022,47.4283828 C91.5710599,47.789395 91.3101934,48.2411837 90.8924842,48.2411837 L74.4582937,48.2411837 Z"/> + <path d="M77.7078135.812538775C77.4981696.45152632 77.7598718 0 78.1787565 0L94.6363581 0C94.8305929 0 95.0100903.10309919 95.1073011.27049884L126.44159 54.22898C126.538963 54.3966589 126.538963 54.6033411 126.44159 54.77102L95.1073011 108.729501C95.0100903 108.896901 94.8305929 109 94.6363581 109L78.1787565 109C77.7598718 109 77.4981696 108.548474 77.7078135 108.187461L108.884718 54.5 77.7078135.812538775zM64.3328086 54.2269735C64.431852 54.3948735 64.4323504 54.6024365 64.3341146 54.7708006L32.850801 108.729282C32.6407695 109.089249 32.1162201 109.090483 31.9044622 108.731508L.0746342025 54.7730265C-.0244091387 54.6051265-.024907567 54.3975635.0733282916 54.2291994L31.5566419.270718284C31.7666733-.0892494466 32.2912227-.0904832685 32.5029806.268492348L64.3328086 54.2269735z"/> + </g> + <path fill="#212529" d="M186.740404,0 L199.453228,0 L199.453228,46.87 L186.740404,46.87 L186.740404,28.5907 L168.674812,28.5907 L168.674812,46.87 L155.961988,46.87 L155.961988,0 L168.674812,0 L168.674812,16.9401571 L186.740404,16.9401571 L186.740404,0 Z M226.217067,0 L238.762617,0 L238.762617,46.87 L226.217067,46.87 L226.217067,0 Z M279.750831,46.87 L261.344607,0.602614286 L261.344607,0 L275.132542,0 L286.644799,31.8046429 L298.157056,0 L311.944991,0 L311.944991,0.602614286 L293.538767,46.87 L279.750831,46.87 Z M346.849018,35.3533714 L373,35.3533714 L373,46.87 L334.108796,46.87 L334.108796,0 L372.530623,0 L372.530623,11.5166286 L346.849018,11.5166286 L346.849018,18.0114714 L363.143091,18.0114714 L363.143091,28.7246143 L346.849018,28.7246143 L346.849018,35.3533714 Z"/> + <path fill="#E31337" d="M175.27868,81.8564453 C178.243977,82.9634821 180.478506,84.538857 181.982335,86.5826172 C183.486164,88.6263774 184.238067,91.0533063 184.238067,93.8634766 C184.238067,97.7806836 182.829573,101.027233 180.012541,103.603223 C177.195509,106.179212 173.520715,107.467187 168.988047,107.467187 L155.961988,107.467187 L155.961988,60.7164062 L165.429709,60.7164062 C170.343629,60.7164062 174.081965,61.7276266 176.644828,63.7500977 C179.207692,65.7725687 180.489104,68.6572078 180.489104,72.4041016 C180.489104,76.491622 178.752314,79.6423717 175.27868,81.8564453 Z M161.311517,79.8765625 L163.507299,79.8765625 C167.382228,79.8765625 170.277618,79.2804747 172.193556,78.0882812 C174.109493,76.8960878 175.067447,75.0013802 175.067447,72.4041016 C175.067447,68.1888461 172.11824,66.08125 166.219736,66.08125 L161.311517,66.08125 L161.311517,79.8765625 Z M161.311517,102.102344 L166.630025,102.102344 C169.959515,102.102344 172.380925,101.780573 173.89433,101.137023 C175.407734,100.493472 176.618439,99.5067099 177.526482,98.1767056 C178.434524,96.8467014 178.888539,95.4952657 178.888539,94.1223581 C178.888539,92.7494504 178.618292,91.5374486 178.077791,90.4863162 C177.537289,89.4351838 176.737359,88.5020496 175.677976,87.6868857 C174.618593,86.8717218 173.321409,86.2603581 171.786384,85.8527761 C170.25136,85.4451942 167.473224,85.2414062 163.451892,85.2414062 L161.311517,85.2414062 L161.311517,102.102344 Z M231.619605,107.467187 L207.928836,107.467187 L207.928836,60.7164062 L213.606789,60.7164062 L213.606789,102.166211 L231.619605,102.166211 L231.619605,107.467187 Z M276.423021,59.95 C280.948221,59.95 285.113879,61.0250869 288.920122,63.175293 C292.726365,65.325499 295.71845,68.2952936 297.896466,72.0847656 C300.074483,75.8742377 301.163475,79.9936301 301.163475,84.4430664 C301.163475,88.8925027 300.085056,93.0118951 297.928185,96.8013672 C295.771314,100.590839 292.810947,103.571278 289.046996,105.742773 C285.283045,107.914269 281.191396,109 276.771925,109 C272.352454,109 268.239659,107.924913 264.433416,105.774707 C260.627174,103.624501 257.645662,100.665351 255.488791,96.897168 C253.33192,93.1289851 252.253501,89.0202371 252.253501,84.5708008 C252.253501,80.1213645 253.33192,75.9913276 255.488791,72.1805664 C257.645662,68.3698052 260.595455,65.3787218 264.338261,63.2072266 C268.081066,61.0357313 272.109279,59.95 276.423021,59.95 Z M276.611833,103.635156 C281.981578,103.635156 286.524315,101.825604 290.240179,98.2064453 C293.956042,94.5872866 295.813947,90.0527616 295.813947,84.6027344 C295.813947,79.1527071 293.934564,74.5756045 290.175742,70.8712891 C286.41692,67.1669737 281.895662,65.3148437 276.611833,65.3148437 C271.199129,65.3148437 266.677872,67.1776181 263.047924,70.9032227 C259.417976,74.6288272 257.603029,79.1633522 257.603029,84.5069336 C257.603029,89.850515 259.471672,94.3743955 263.209015,98.0787109 C266.946358,101.783026 271.413919,103.635156 276.611833,103.635156 Z M373,84.9859375 C372.87263,92.5648816 370.675524,98.461893 366.408617,102.677148 C362.14171,106.892404 356.526884,109 349.563971,109 C341.582095,109 335.213672,106.541138 330.458512,101.62334 C325.703352,96.7055418 323.325807,90.946908 323.325807,84.3472656 C323.325807,77.7476233 325.703352,72.0315671 330.458512,67.1989258 C335.213672,62.3662844 341.348586,59.95 348.863437,59.95 C352.939289,59.95 356.7285,60.7057542 360.231185,62.2172852 C363.73387,63.7288162 367.162204,66.0386563 370.51629,69.146875 L366.313089,73.1705078 C363.723225,70.6583859 360.921119,68.731745 357.906688,67.3905273 C354.892256,66.0493097 351.792957,65.3787109 348.608698,65.3787109 C345.424439,65.3787109 342.261455,66.2302649 339.119653,67.9333984 C335.977851,69.636532 333.515394,71.9357277 331.732209,74.8310547 C329.949024,77.7263817 329.057445,80.8558426 329.057445,84.2195313 C329.057445,89.4566668 331.021042,94.0231251 334.948295,97.919043 C338.875547,101.814961 343.789847,103.762891 349.69134,103.762891 C354.064389,103.762891 357.842987,102.560071 361.027246,100.154395 C364.211505,97.7487184 366.164488,94.4595912 366.886253,90.2869141 L352.366105,90.2869141 L352.366105,84.9859375 L373,84.9859375 Z"/> + </g> +</svg> diff --git a/src/app/assets/images/steemit.svg b/src/app/assets/images/steemit.svg index fb0675bfceedfaddde5b2fcae8e45c4ab48ef35c..d4e160862519e20f412993fe8f2c8e7607a2fbb2 100644 --- a/src/app/assets/images/steemit.svg +++ b/src/app/assets/images/steemit.svg @@ -1,15 +1,10 @@ -<?xml version="1.0" encoding="UTF-8" standalone="no"?> -<!-- Generator: GIMP export as svg plugin --> - -<svg xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:cc="http://creativecommons.org/ns#" xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" xmlns:svg="http://www.w3.org/2000/svg" xmlns="http://www.w3.org/2000/svg" xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd" xmlns:xlink="http://www.w3.org/1999/xlink" xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape" width="373px" height="109px" viewBox="0 0 373 109" version="1.1"><defs id="defs2" /> -<sodipodi:namedview id="base" pagecolor="#ffffff" bordercolor="#666666" borderopacity="1.0" inkscape:pageopacity="0.0" inkscape:pageshadow="2" inkscape:document-units="px" showgrid="false" inkscape:window-maximized="1" /> -<metadata id="metadata5"> -<rdf:RDF> -<cc:Work rdf:about=""> -<dc:format>image/svg+xml</dc:format> -<dc:type rdf:resource="http://purl.org/dc/dcmitype/StillImage" /> -<dc:title></dc:title> -</cc:Work> -</rdf:RDF> -</metadata><g inkscape:groupmode="layer" inkscape:label="hive-blog-logo.png" ><image xlink:href="" x="0" y="0" width="373" height="109" /> -</g></svg> \ No newline at end of file +<svg xmlns="http://www.w3.org/2000/svg" width="373" height="109" viewBox="0 0 373 109"> + <g fill="none"> + <g fill="#E31337"> + <path d="M90.4422108,61.5356296 C90.85992,61.5356296 91.1207865,61.9874183 90.9115288,62.3484305 L64.0269732,108.729763 C63.930027,108.897015 63.7511741,109 63.5576553,109 L47.1234647,109 C46.7057556,109 46.4448891,108.548211 46.6541468,108.187199 L73.5387023,61.8058664 C73.6356485,61.6386145 73.8145015,61.5356296 74.0080203,61.5356296 L90.4422108,61.5356296 Z M74.4582937,48.2411837 C74.2647749,48.2411837 74.0859219,48.1381988 73.9889757,47.9709469 L46.6541468,0.812800858 C46.4448891,0.451788656 46.7057556,0 47.1234647,0 L63.5576553,0 C63.7511741,0 63.930027,0.102984848 64.0269732,0.270236757 L91.3618022,47.4283828 C91.5710599,47.789395 91.3101934,48.2411837 90.8924842,48.2411837 L74.4582937,48.2411837 Z"/> + <path d="M77.7078135.812538775C77.4981696.45152632 77.7598718 0 78.1787565 0L94.6363581 0C94.8305929 0 95.0100903.10309919 95.1073011.27049884L126.44159 54.22898C126.538963 54.3966589 126.538963 54.6033411 126.44159 54.77102L95.1073011 108.729501C95.0100903 108.896901 94.8305929 109 94.6363581 109L78.1787565 109C77.7598718 109 77.4981696 108.548474 77.7078135 108.187461L108.884718 54.5 77.7078135.812538775zM64.3328086 54.2269735C64.431852 54.3948735 64.4323504 54.6024365 64.3341146 54.7708006L32.850801 108.729282C32.6407695 109.089249 32.1162201 109.090483 31.9044622 108.731508L.0746342025 54.7730265C-.0244091387 54.6051265-.024907567 54.3975635.0733282916 54.2291994L31.5566419.270718284C31.7666733-.0892494466 32.2912227-.0904832685 32.5029806.268492348L64.3328086 54.2269735z"/> + </g> + <path fill="#212529" d="M186.740404,0 L199.453228,0 L199.453228,46.87 L186.740404,46.87 L186.740404,28.5907 L168.674812,28.5907 L168.674812,46.87 L155.961988,46.87 L155.961988,0 L168.674812,0 L168.674812,16.9401571 L186.740404,16.9401571 L186.740404,0 Z M226.217067,0 L238.762617,0 L238.762617,46.87 L226.217067,46.87 L226.217067,0 Z M279.750831,46.87 L261.344607,0.602614286 L261.344607,0 L275.132542,0 L286.644799,31.8046429 L298.157056,0 L311.944991,0 L311.944991,0.602614286 L293.538767,46.87 L279.750831,46.87 Z M346.849018,35.3533714 L373,35.3533714 L373,46.87 L334.108796,46.87 L334.108796,0 L372.530623,0 L372.530623,11.5166286 L346.849018,11.5166286 L346.849018,18.0114714 L363.143091,18.0114714 L363.143091,28.7246143 L346.849018,28.7246143 L346.849018,35.3533714 Z"/> + <path fill="#E31337" d="M175.27868,81.8564453 C178.243977,82.9634821 180.478506,84.538857 181.982335,86.5826172 C183.486164,88.6263774 184.238067,91.0533063 184.238067,93.8634766 C184.238067,97.7806836 182.829573,101.027233 180.012541,103.603223 C177.195509,106.179212 173.520715,107.467187 168.988047,107.467187 L155.961988,107.467187 L155.961988,60.7164062 L165.429709,60.7164062 C170.343629,60.7164062 174.081965,61.7276266 176.644828,63.7500977 C179.207692,65.7725687 180.489104,68.6572078 180.489104,72.4041016 C180.489104,76.491622 178.752314,79.6423717 175.27868,81.8564453 Z M161.311517,79.8765625 L163.507299,79.8765625 C167.382228,79.8765625 170.277618,79.2804747 172.193556,78.0882812 C174.109493,76.8960878 175.067447,75.0013802 175.067447,72.4041016 C175.067447,68.1888461 172.11824,66.08125 166.219736,66.08125 L161.311517,66.08125 L161.311517,79.8765625 Z M161.311517,102.102344 L166.630025,102.102344 C169.959515,102.102344 172.380925,101.780573 173.89433,101.137023 C175.407734,100.493472 176.618439,99.5067099 177.526482,98.1767056 C178.434524,96.8467014 178.888539,95.4952657 178.888539,94.1223581 C178.888539,92.7494504 178.618292,91.5374486 178.077791,90.4863162 C177.537289,89.4351838 176.737359,88.5020496 175.677976,87.6868857 C174.618593,86.8717218 173.321409,86.2603581 171.786384,85.8527761 C170.25136,85.4451942 167.473224,85.2414062 163.451892,85.2414062 L161.311517,85.2414062 L161.311517,102.102344 Z M231.619605,107.467187 L207.928836,107.467187 L207.928836,60.7164062 L213.606789,60.7164062 L213.606789,102.166211 L231.619605,102.166211 L231.619605,107.467187 Z M276.423021,59.95 C280.948221,59.95 285.113879,61.0250869 288.920122,63.175293 C292.726365,65.325499 295.71845,68.2952936 297.896466,72.0847656 C300.074483,75.8742377 301.163475,79.9936301 301.163475,84.4430664 C301.163475,88.8925027 300.085056,93.0118951 297.928185,96.8013672 C295.771314,100.590839 292.810947,103.571278 289.046996,105.742773 C285.283045,107.914269 281.191396,109 276.771925,109 C272.352454,109 268.239659,107.924913 264.433416,105.774707 C260.627174,103.624501 257.645662,100.665351 255.488791,96.897168 C253.33192,93.1289851 252.253501,89.0202371 252.253501,84.5708008 C252.253501,80.1213645 253.33192,75.9913276 255.488791,72.1805664 C257.645662,68.3698052 260.595455,65.3787218 264.338261,63.2072266 C268.081066,61.0357313 272.109279,59.95 276.423021,59.95 Z M276.611833,103.635156 C281.981578,103.635156 286.524315,101.825604 290.240179,98.2064453 C293.956042,94.5872866 295.813947,90.0527616 295.813947,84.6027344 C295.813947,79.1527071 293.934564,74.5756045 290.175742,70.8712891 C286.41692,67.1669737 281.895662,65.3148437 276.611833,65.3148437 C271.199129,65.3148437 266.677872,67.1776181 263.047924,70.9032227 C259.417976,74.6288272 257.603029,79.1633522 257.603029,84.5069336 C257.603029,89.850515 259.471672,94.3743955 263.209015,98.0787109 C266.946358,101.783026 271.413919,103.635156 276.611833,103.635156 Z M373,84.9859375 C372.87263,92.5648816 370.675524,98.461893 366.408617,102.677148 C362.14171,106.892404 356.526884,109 349.563971,109 C341.582095,109 335.213672,106.541138 330.458512,101.62334 C325.703352,96.7055418 323.325807,90.946908 323.325807,84.3472656 C323.325807,77.7476233 325.703352,72.0315671 330.458512,67.1989258 C335.213672,62.3662844 341.348586,59.95 348.863437,59.95 C352.939289,59.95 356.7285,60.7057542 360.231185,62.2172852 C363.73387,63.7288162 367.162204,66.0386563 370.51629,69.146875 L366.313089,73.1705078 C363.723225,70.6583859 360.921119,68.731745 357.906688,67.3905273 C354.892256,66.0493097 351.792957,65.3787109 348.608698,65.3787109 C345.424439,65.3787109 342.261455,66.2302649 339.119653,67.9333984 C335.977851,69.636532 333.515394,71.9357277 331.732209,74.8310547 C329.949024,77.7263817 329.057445,80.8558426 329.057445,84.2195313 C329.057445,89.4566668 331.021042,94.0231251 334.948295,97.919043 C338.875547,101.814961 343.789847,103.762891 349.69134,103.762891 C354.064389,103.762891 357.842987,102.560071 361.027246,100.154395 C364.211505,97.7487184 366.164488,94.4595912 366.886253,90.2869141 L352.366105,90.2869141 L352.366105,84.9859375 L373,84.9859375 Z"/> + </g> +</svg> diff --git a/src/app/assets/stylesheets/_themes.scss b/src/app/assets/stylesheets/_themes.scss index f85fa82eb2b96926941772ddf7d70df61829a5d5..6b0e79e6d73fd8da9e9283a09914ffb06ac0dbea 100755 --- a/src/app/assets/stylesheets/_themes.scss +++ b/src/app/assets/stylesheets/_themes.scss @@ -44,7 +44,7 @@ $themes: ( colorAccentHover: $color-hive-red-dark, colorAccentReverse: $color-hive-black, colorWhite: $color-white, - backgroundColor: $color-background-off-white, + backgroundColor: $color-background-off-white-dark, backgroundColorEmphasis: $color-background-almost-white, backgroundColorOpaque: $color-background-off-white, backgroundTransparent: transparent, @@ -53,6 +53,7 @@ $themes: ( moduleMediumBackgroundColor: $color-transparent, navBackgroundColor: $color-white, highlightBackgroundColor: #f3faf0, + tableRowOddBackgroundColor: #e5e5e5, tableRowEvenBackgroundColor: #f4f4f4, border: 1px solid $color-border-light, borderLight: 1px solid $color-border-light-lightest, @@ -94,6 +95,7 @@ $themes: ( moduleMediumBackgroundColor: $color-background-dark, navBackgroundColor: $color-background-less-dark, highlightBackgroundColor: $color-hive-black-darkest, + tableRowOddBackgroundColor: #283239, tableRowEvenBackgroundColor: #212C33, border: 1px solid $color-border-dark-lightest, borderLight: 1px solid $color-border-dark-lightest, diff --git a/src/app/assets/stylesheets/_variables.scss b/src/app/assets/stylesheets/_variables.scss index 5fbd540876874174442c5b8e774d419e0f383e1c..4b97d5c3242a651063055023ab5bf5fcb2933abe 100755 --- a/src/app/assets/stylesheets/_variables.scss +++ b/src/app/assets/stylesheets/_variables.scss @@ -19,7 +19,8 @@ $color-transparent: transparent; $color-background-almost-white:#fafaff; $color-background-off-white: #f8f8ff; -$color-background-off-white-light: #f4f4fd; +$color-background-off-white-dark: #f4f4fd; +$color-background-off-white-darker: #f0f0f09; $color-background-dark: $color-hive-black; $color-background-less-dark: #2c3136; $color-background-super-dark: #191c1f; diff --git a/src/app/components/cards/MarkdownViewer.jsx b/src/app/components/cards/MarkdownViewer.jsx index dda19136d355014484fff83bd5c3280c738f1b67..beae2075b37e22a1b42ae6de0f8cbaadb9884e58 100644 --- a/src/app/components/cards/MarkdownViewer.jsx +++ b/src/app/components/cards/MarkdownViewer.jsx @@ -3,11 +3,11 @@ import PropTypes from 'prop-types'; import { connect } from 'react-redux'; import { Component } from 'react'; import Remarkable from 'remarkable'; -import YoutubePreview from 'app/components/elements/YoutubePreview'; import sanitizeConfig, { noImageText } from 'app/utils/SanitizeConfig'; import sanitize from 'sanitize-html'; import HtmlReady from 'shared/HtmlReady'; import tt from 'counterpart'; +import { generateMd as EmbeddedPlayerGenerateMd } from 'app/components/elements/EmbeddedPlayers'; const remarkable = new Remarkable({ html: true, // remarkable renders first then sanitize runs... @@ -136,114 +136,37 @@ class MarkdownViewer extends Component { let idx = 0; const sections = []; + function checksum(s) { + let chk = 0x12345678; + const len = s.length; + for (let i = 0; i < len; i += 1) { + chk += s.charCodeAt(i) * (i + 1); + } + + return (chk & 0xffffffff).toString(16); + } + // HtmlReady inserts ~~~ embed:${id} type ~~~ for (let section of cleanText.split('~~~ embed:')) { - const match = section.match( - /^([A-Za-z0-9\?\=\_\-\/\.]+) (youtube|vimeo|twitch|dtube|threespeak)\s?(\d+)? ~~~/ - ); - if (match && match.length >= 3) { - const id = match[1]; - const type = match[2]; - const startTime = match[3] ? parseInt(match[3]) : 0; - const w = large ? 640 : 480, - h = large ? 360 : 270; - - if (type === 'youtube') { - sections.push( - <YoutubePreview - key={id} - width={w} - height={h} - youTubeId={id} - startTime={startTime} - frameBorder="0" - allowFullScreen="true" - /> - ); - } else if (type === 'threespeak') { - const url = `https://3speak.online/embed?v=${id}`; - sections.push( - <div className="videoWrapper" key={id}> - <iframe - src={url} - width={w} - height={h} - frameBorder="0" - webkitallowfullscreen="true" - mozallowfullscreen="true" - allowFullScreen - title={`ThreeSpeak video ${id}`} - /> - </div> - ); - } else if (type === 'vimeo') { - const url = `https://player.vimeo.com/video/${id}#t=${ - startTime - }s`; - sections.push( - <div className="videoWrapper" key={id}> - <iframe - src={url} - width={w} - height={h} - frameBorder="0" - webkitallowfullscreen="true" - mozallowfullscreen="true" - allowFullScreen - title={`Vimeo video ${id}`} - /> - </div> - ); - } else if (type === 'twitch') { - const url = `https://player.twitch.tv/${id}`; - sections.push( - <div className="videoWrapper" key={id}> - <iframe - src={url} - width={w} - height={h} - frameBorder="0" - webkitallowfullscreen="true" - mozallowfullscreen="true" - allowFullScreen - title={`Twitch video ${id}`} - /> - </div> - ); - } else if (type === 'dtube') { - const url = `https://emb.d.tube/#!/${id}`; - sections.push( - <div className="videoWrapper" key={id}> - <iframe - src={url} - width={w} - height={h} - frameBorder="0" - webkitallowfullscreen="true" - mozallowfullscreen="true" - allowFullScreen - title={`DTube video ${id}`} - /> - </div> - ); - } else { - console.error('MarkdownViewer unknown embed type', type); - } - if (match[3]) { - section = section.substring( - `${id} ${type} ${startTime} ~~~`.length - ); - } else { - section = section.substring(`${id} ${type} ~~~`.length); + const embedMd = EmbeddedPlayerGenerateMd(section, idx, large); + if (embedMd) { + const { section: newSection, markdown } = embedMd; + section = newSection; + sections.push(markdown); + + if (section === '') { + continue; } - if (section === '') continue; } + sections.push( <div - key={idx++} + key={checksum(section)} dangerouslySetInnerHTML={{ __html: section }} /> ); + + idx += 1; } const cn = @@ -251,12 +174,14 @@ class MarkdownViewer extends Component { (this.props.className ? ` ${this.props.className}` : '') + (html ? ' html' : '') + (large ? '' : ' MarkdownViewer--small'); + return ( <div className={'MarkdownViewer ' + cn}> {sections} {noImageActive && allowNoImage && ( <div + key={'hidden-content'} onClick={this.onAllowNoImage} className="MarkdownViewer__negative_group" > diff --git a/src/app/components/cards/NotificationsList.jsx b/src/app/components/cards/NotificationsList.jsx index 2ac07d91b9f9a7802bd500cbb471db84ef719d14..7228dff74d560c0ac2077d3b93aee3f65de653e5 100644 --- a/src/app/components/cards/NotificationsList.jsx +++ b/src/app/components/cards/NotificationsList.jsx @@ -133,11 +133,19 @@ class NotificationsList extends React.Component { 'notification__item' ); + let visibleCount = 0; for (let ni = 0; ni < notificationElements.length; ni += 1) { const notificationElement = notificationElements[ni]; if (notificationFilter === 'all') { notificationElement.classList.remove('hide'); + + visibleCount += 1; + if (visibleCount % 2 === 0) { + notificationElement.classList.add('even'); + } else { + notificationElement.classList.remove('even'); + } } else if ( this.notificationFilterToTypes.hasOwnProperty( notificationFilter @@ -163,6 +171,13 @@ class NotificationsList extends React.Component { notificationElement.classList.add('hide'); } else { notificationElement.classList.remove('hide'); + visibleCount += 1; + } + + if (visibleCount % 2 === 0) { + notificationElement.classList.add('even'); + } else { + notificationElement.classList.remove('even'); } } } @@ -218,10 +233,13 @@ class NotificationsList extends React.Component { className={`notification__item flex-body notification__${ item.type }`} - style={{ - background: 'rgba(225,255,225,' + item.score + '%)', - }} > + <div className="notification__score"> + <div + className="notification__score_bar" + style={{ width: `${item.score}%` }} + /> + </div> <div className="flex-row"> {mentions && participants && participants[0]} </div> diff --git a/src/app/components/cards/NotificationsList.scss b/src/app/components/cards/NotificationsList.scss index 024ea326006c68f2c1de19c8335969f76e20a714..89e516f3769f2ec2502cbd74c3e1a8bbffb7e25a 100644 --- a/src/app/components/cards/NotificationsList.scss +++ b/src/app/components/cards/NotificationsList.scss @@ -29,17 +29,24 @@ font-weight: bold; } +.notification__item.even { + @include themify($themes) { + background-color: themed('tableRowEvenBackgroundColor'); + } +} + .notification__item { + @include themify($themes) { + background-color: themed('tableRowOddBackgroundColor'); + } align-items: center; padding: 0.5em 1rem; + margin-bottom: 2px; position: relative; - @include themify($themes) { - border-bottom: themed('border'); - } .notification__unread { position: absolute; - right: 1em; + left: 0.4em; top: 0.75em; font-size: 2em; @include themify($themes) { @@ -76,5 +83,32 @@ padding-bottom: 0.5em; padding-right: 0.5em; } + + .notification__message a { + @include themify($themes) { + color: themed('textColorPrimary'); + } + } + + .notification__message a:visited { + @include themify($themes) { + color: themed('textColorSecondary'); + } + } + + .notification__score { + position: absolute; + top: 5px; + right: 5px; + height: 4px; + width: 35px; + background-color: lightgray; + } + + .notification__score_bar { + height: 100%; + position: relative; + background-color: $color-text-hive-red; + } } diff --git a/src/app/components/cards/PostTemplateSelector.jsx b/src/app/components/cards/PostTemplateSelector.jsx new file mode 100644 index 0000000000000000000000000000000000000000..3e49cbfdb158e24b3b4d081d414ebd4ba15177c0 --- /dev/null +++ b/src/app/components/cards/PostTemplateSelector.jsx @@ -0,0 +1,96 @@ +import React from 'react'; +import tt from 'counterpart'; + +class PostTemplateSelector extends React.Component { + static propTypes = { + username: React.PropTypes.string.isRequired, + templates: React.PropTypes.array.isRequired, + onChange: React.PropTypes.func.isRequired, + }; + + constructor() { + super(); + this.state = { + currentTemplateName: '', + }; + } + + render() { + const { username, onChange, templates } = this.props; + const { currentTemplateName } = this.state; + if (!username || typeof window === 'undefined') { + return null; + } + + const handleTemplateSelection = (event, create = false) => { + const selectedTemplateName = event.target.value; + this.setState({ currentTemplateName: selectedTemplateName }); + onChange( + create ? `create_${selectedTemplateName}` : selectedTemplateName + ); + }; + + return ( + <div> + <div className="row"> + <div className="column"> + <h4>{tt('post_template_selector_jsx.templates')}</h4> + <p> + {tt( + 'post_template_selector_jsx.templates_description' + )} + </p> + </div> + </div> + <div className="row"> + <div className="small-12 medium-6 large-12 columns"> + {templates && ( + <select + onChange={handleTemplateSelection} + value={currentTemplateName} + > + <option value=""> + {tt( + 'post_template_selector_jsx.choose_template' + )} + </option> + {templates.map(template => ( + <option + value={template.name} + key={template.name} + > + {template.name} + </option> + ))} + </select> + )} + {!templates && ( + <span> + {tt( + 'post_template_selector_jsx.create_template_first' + )} + </span> + )} + </div> + </div> + <div className="row"> + <div className="small-12 medium-6 large-12 columns"> + <input + id="new_template_name" + type="text" + className="input-group-field bold" + placeholder={tt( + 'post_template_selector_jsx.new_template_name' + )} + onChange={event => { + handleTemplateSelection(event, true); + }} + /> + </div> + </div> + </div> + ); + } +} + +export default PostTemplateSelector; diff --git a/src/app/components/elements/EmbeddedPlayers/dtube.jsx b/src/app/components/elements/EmbeddedPlayers/dtube.jsx new file mode 100644 index 0000000000000000000000000000000000000000..61d5d7d2b35ffeee012d9df3067ed9dbccf8a1a2 --- /dev/null +++ b/src/app/components/elements/EmbeddedPlayers/dtube.jsx @@ -0,0 +1,109 @@ +import React from 'react'; + +/** + * Regular expressions for detecting and validating provider URLs + * @type {{htmlReplacement: RegExp, main: RegExp, sanitize: RegExp}} + */ +const regex = { + sanitize: /^https:\/\/emb.d.tube\/\#\!\/([a-zA-Z0-9\-\.\/]+)$/, + main: /https:\/\/(?:emb\.)?(?:d.tube\/\#\!\/(?:v\/)?)([a-zA-Z0-9\-\.\/]*)/, + contentId: /(?:d\.tube\/#!\/(?:v\/)?([a-zA-Z0-9\-\.\/]*))+/, +}; + +export default regex; + +/** + * Generates the Markdown/HTML code to override the detected URL with an iFrame + * @param idx + * @param threespeakId + * @param w + * @param h + * @returns {*} + */ +export function genIframeMd(idx, dtubeId, w, h) { + const url = `https://emb.d.tube/#!/${dtubeId}`; + return ( + <div key={`dtube-${dtubeId}-${idx}`} className="videoWrapper"> + <iframe + title="DTube embedded player" + key={idx} + src={url} + width={w} + height={h} + frameBorder="0" + allowFullScreen + /> + </div> + ); +} + +/** + * Check if the iframe code in the post editor is to an allowed URL + * <iframe title="DTube embedded player" src="https://emb.d.tube/#!/lemwong/QmQqxBCkoVusMRwP6D9oBMRQdASFzABdKQxE7xLysfmsR6" width="640" height="360" frameborder="0" allowfullscreen=""></iframe> + * @param url + * @returns {boolean|*} + */ +export function validateIframeUrl(url) { + const match = url.match(regex.sanitize); + + if (match) { + return url; + } + + return false; +} + +/** + * Rewrites the embedded URL to a normalized format + * @param url + * @returns {string|boolean} + */ +export function normalizeEmbedUrl(url) { + const match = url.match(regex.contentId); + + if (match && match.length >= 2) { + return `https://emb.d.tube/#!/${match[1]}`; + } + + return false; +} + +/** + * Extract the content ID and other metadata from the URL + * @param data + * @returns {null|{id: *, canonical: string, url: *}} + */ +function extractContentId(data) { + if (!data) return null; + + const m = data.match(regex.main); + if (!m || m.length < 2) return null; + + return { + id: m[1], + url: m[0], + canonical: `https://emb.d.tube/#!/${m[1]}`, + }; +} + +/** + * Replaces the URL with a custom Markdown for embedded players + * @param child + * @param links + * @returns {*} + */ +export function embedNode(child, links /*images*/) { + try { + const data = child.data; + const dtube = extractContentId(data); + if (!dtube) return child; + + child.data = data.replace(dtube.url, `~~~ embed:${dtube.id} dtube ~~~`); + + if (links) links.add(dtube.canonical); + } catch (error) { + console.log(error); + } + + return child; +} diff --git a/src/app/components/elements/EmbeddedPlayers/index.jsx b/src/app/components/elements/EmbeddedPlayers/index.jsx new file mode 100644 index 0000000000000000000000000000000000000000..c69f4e087ac7b7c9714b77e9e7068527e92b45a2 --- /dev/null +++ b/src/app/components/elements/EmbeddedPlayers/index.jsx @@ -0,0 +1,230 @@ +import { + genIframeMd as genDtubeIframeMd, + validateIframeUrl as validateDtubeIframeUrl, + normalizeEmbedUrl as normalizeDtubeEmbedUrl, + embedNode as embedDtubeNode, +} from 'app/components/elements/EmbeddedPlayers/dtube'; + +import { + genIframeMd as genTwitchIframeMd, + validateIframeUrl as validateTwitchIframeUrl, + normalizeEmbedUrl as normalizeTwitchEmbedUrl, + embedNode as embedTwitchNode, +} from 'app/components/elements/EmbeddedPlayers/twitch'; + +import { validateIframeUrl as validateSoundcloudIframeUrl } from 'app/components/elements/EmbeddedPlayers/soundcloud'; + +import { + genIframeMd as genYoutubeIframeMd, + validateIframeUrl as validateYoutubeIframeUrl, + normalizeEmbedUrl as normalizeYoutubeEmbedUrl, + embedNode as embedYoutubeNode, +} from 'app/components/elements/EmbeddedPlayers/youtube'; + +import { + genIframeMd as genVimeoIframeMd, + validateIframeUrl as validateVimeoIframeUrl, + normalizeEmbedUrl as normalizeVimeoEmbedUrl, + embedNode as embedVimeoNode, +} from 'app/components/elements/EmbeddedPlayers/vimeo'; + +import { + genIframeMd as genThreespeakIframeMd, + validateIframeUrl as validateThreespeakIframeUrl, + normalizeEmbedUrl as normalizeThreespeakEmbedUrl, + embedNode as embedThreeSpeakNode, + preprocessHtml as preprocess3SpeakHtml, +} from 'app/components/elements/EmbeddedPlayers/threespeak'; + +const supportedProviders = [ + { + id: 'dtube', + validateIframeUrlFn: validateDtubeIframeUrl, + normalizeEmbedUrlFn: normalizeDtubeEmbedUrl, + embedNodeFn: embedDtubeNode, + genIframeMdFn: genDtubeIframeMd, + }, + { + id: 'twitch', + validateIframeUrlFn: validateTwitchIframeUrl, + normalizeEmbedUrlFn: normalizeTwitchEmbedUrl, + embedNodeFn: embedTwitchNode, + genIframeMdFn: genTwitchIframeMd, + }, + { + id: 'soundcloud', + validateIframeUrlFn: validateSoundcloudIframeUrl, + normalizeEmbedUrlFn: null, + embedNodeFn: null, + genIframeMdFn: null, + }, + { + id: 'youtube', + validateIframeUrlFn: validateYoutubeIframeUrl, + normalizeEmbedUrlFn: normalizeYoutubeEmbedUrl, + embedNodeFn: embedYoutubeNode, + genIframeMdFn: genYoutubeIframeMd, + }, + { + id: 'vimeo', + validateIframeUrlFn: validateVimeoIframeUrl, + normalizeEmbedUrlFn: normalizeVimeoEmbedUrl, + embedNodeFn: embedVimeoNode, + genIframeMdFn: genVimeoIframeMd, + }, + { + id: 'threespeak', + validateIframeUrlFn: validateThreespeakIframeUrl, + normalizeEmbedUrlFn: normalizeThreespeakEmbedUrl, + embedNodeFn: embedThreeSpeakNode, + genIframeMdFn: genThreespeakIframeMd, + }, +]; + +export default supportedProviders; + +/** + * Allow iFrame in the Markdown if the source URL is allowed + * @param url + * @returns {boolean|*} + */ +export function validateIframeUrl(url) { + for (let pi = 0; pi < supportedProviders.length; pi += 1) { + const provider = supportedProviders[pi]; + + const validIframeUrl = provider.validateIframeUrlFn(url); + + if (validIframeUrl !== false) { + console.log(`Found a valid ${provider.id} iframe URL`); + return validIframeUrl; + } + } + + return false; +} + +/** + * Rewrites the embedded URL to a normalized format + * @param url + * @returns {boolean|*} + */ +export function normalizeEmbedUrl(url) { + for (let pi = 0; pi < supportedProviders.length; pi += 1) { + const provider = supportedProviders[pi]; + + if (typeof provider.normalizeEmbedUrlFn === 'function') { + const validEmbedUrl = provider.normalizeEmbedUrlFn(url); + + if (validEmbedUrl !== false) { + console.log(`Found a valid ${provider.id} embedded URL`); + return validEmbedUrl; + } + } + } + + return false; +} + +/** + * Replaces the URL with a custom Markdown for embedded players + * @param child + * @param links + * @param images + * @returns {*} + */ +export function embedNode(child, links, images) { + for (let pi = 0; pi < supportedProviders.length; pi += 1) { + const provider = supportedProviders[pi]; + + if (typeof provider.embedNodeFn === 'function') { + child = provider.embedNodeFn(child, links, images); + } + } + + return child; +} + +/** + * Returns the provider config by ID + * @param id + * @returns {null|{normalizeEmbedUrlFn, validateIframeUrlFn, id: string, genIframeMdFn, embedNodeFn}|{normalizeEmbedUrlFn, validateIframeUrlFn, id: string, genIframeMdFn, embedNodeFn}|{normalizeEmbedUrlFn: null, validateIframeUrlFn, id: string, genIframeMdFn: null, embedNodeFn: null}|{normalizeEmbedUrlFn, validateIframeUrlFn, id: string, genIframeMdFn, embedNodeFn}|{normalizeEmbedUrlFn, validateIframeUrlFn, id: string, genIframeMdFn, embedNodeFn}} + */ +function getProviderById(id) { + for (let pi = 0; pi < supportedProviders.length; pi += 1) { + const provider = supportedProviders[pi]; + + if (provider.id === id) { + return provider; + } + } + + return null; +} + +/** + * Returns all providers IDs + * @returns {(string)[]} + */ +function getProviderIds() { + return supportedProviders.map(o => { + return o.id; + }); +} + +/** + * Replaces ~~~ embed: Markdown code to an iframe MD + * @param section + * @param idx + * @param large + * @returns {null|{markdown: null, section: string}} + */ +export function generateMd(section, idx, large) { + let markdown = null; + const supportedProvidersIds = getProviderIds(); + const regex = new RegExp( + `^([A-Za-z0-9\\?\\=\\_\\-\\/\\.]+) (${supportedProvidersIds.join( + '|' + )})\\s?(\\d+)? ~~~` + ); + const match = section.match(regex); + + if (match && match.length >= 3) { + const id = match[1]; + const type = match[2]; + const startTime = match[3] ? parseInt(match[3]) : 0; + const w = large ? 640 : 480, + h = large ? 360 : 270; + + const provider = getProviderById(type); + if (provider) { + markdown = provider.genIframeMdFn(idx, id, w, h, startTime); + } else { + console.error('MarkdownViewer unknown embed type', type); + } + + if (match[3]) { + section = section.substring( + `${id} ${type} ${startTime} ~~~`.length + ); + } else { + section = section.substring(`${id} ${type} ~~~`.length); + } + + return { + section, + markdown, + }; + } + + return null; +} + +/** + * Pre-process HTML codes from the Markdown before it gets transformed + * @param html + * @returns {*} + */ +export function preprocessHtml(html) { + html = preprocess3SpeakHtml(html); + return html; +} diff --git a/src/app/components/elements/EmbeddedPlayers/soundcloud.jsx b/src/app/components/elements/EmbeddedPlayers/soundcloud.jsx new file mode 100644 index 0000000000000000000000000000000000000000..2a72f6c32807e69f2474466ce12c0dde98c4ade0 --- /dev/null +++ b/src/app/components/elements/EmbeddedPlayers/soundcloud.jsx @@ -0,0 +1,27 @@ +/** + * Regular expressions for detecting and validating provider URLs + * @type {{htmlReplacement: RegExp, main: RegExp, sanitize: RegExp}} + */ +const regex = { + sanitize: /^https:\/\/w.soundcloud.com\/player\/.*?url=(.+?)&.*/i, +}; + +export default regex; + +/** + * Check if the iframe code in the post editor is to an allowed URL + * <iframe width="100%" height="450" scrolling="no" frameborder="no" src="https://w.soundcloud.com/player/?url=https%3A//api.soundcloud.com/tracks/257659076&auto_play=false&hide_related=false&show_comments=true&show_user=true&show_reposts=false&visual=true"></iframe> + * @param url + * @returns {boolean|*} + */ +export function validateIframeUrl(url) { + const match = url.match(regex.sanitize); + + if (!match || match.length !== 2) { + return false; + } + + return `https://w.soundcloud.com/player/?url=${ + match[1] + }&auto_play=false&hide_related=false&show_comments=true&show_user=true&show_reposts=false&visual=true`; +} diff --git a/src/app/components/elements/EmbeddedPlayers/threespeak.jsx b/src/app/components/elements/EmbeddedPlayers/threespeak.jsx new file mode 100644 index 0000000000000000000000000000000000000000..448dbc440a4977e2b307d56f1f05eca78cd44616 --- /dev/null +++ b/src/app/components/elements/EmbeddedPlayers/threespeak.jsx @@ -0,0 +1,140 @@ +import React from 'react'; + +/** + * Regular expressions for detecting and validating provider URLs + * @type {{htmlReplacement: RegExp, main: RegExp, sanitize: RegExp}} + */ +const regex = { + sanitize: /^https:\/\/3speak.online\/embed\?v=([A-Za-z0-9\_\-\/]+)(&.*)?$/, + main: /(?:https?:\/\/(?:(?:3speak.online\/watch\?v=)|(?:3speak.online\/embed\?v=)))([A-Za-z0-9\_\-\/]+)(&.*)?/i, + htmlReplacement: /<a href="(https?:\/\/3speak.online\/watch\?v=([A-Za-z0-9\_\-\/]+))".*<img.*?><\/a>/i, +}; + +export default regex; + +/** + * Generates the Markdown/HTML code to override the detected URL with an iFrame + * @param idx + * @param threespeakId + * @param w + * @param h + * @returns {*} + */ +export function genIframeMd(idx, threespeakId, w, h) { + const url = `https://3speak.online/embed?v=${threespeakId}`; + return ( + <div key={`threespeak-${threespeakId}-${idx}`} className="videoWrapper"> + <iframe + title="3Speak embedded player" + key={idx} + src={url} + width={w} + height={h} + frameBorder="0" + allowFullScreen + /> + </div> + ); +} + +/** + * Check if the iframe code in the post editor is to an allowed URL + * @param url + * @returns {boolean|*} + */ +export function validateIframeUrl(url) { + const match = url.match(regex.sanitize); + + if (match) { + return url; + } + + return false; +} + +/** + * Rewrites the embedded URL to a normalized format + * @param url + * @returns {string|boolean} + */ +export function normalizeEmbedUrl(url) { + const match = url.match(regex.contentId); + + if (match && match.length >= 2) { + return `https://3speak.online/embed?v=${match[1]}`; + } + + return false; +} + +/** + * Extract the content ID and other metadata from the URL + * @param data + * @returns {null|{id: *, canonical: string, url: *}} + */ +function extractContentId(data) { + if (!data) return null; + + const match = data.match(regex.main); + const url = match ? match[0] : null; + if (!url) return null; + const fullId = match[1]; + const id = fullId.split('/').pop(); + + return { + id, + fullId, + url, + thumbnail: `https://img.3speakcontent.online/${id}/post.png`, + }; +} + +/** + * Replaces the URL with a custom Markdown for embedded players + * @param child + * @param links + * @returns {*} + */ +export function embedNode(child, links /*images*/) { + try { + const data = child.data; + const threespeak = extractContentId(data); + if (!threespeak) return child; + + child.data = data.replace( + threespeak.url, + `~~~ embed:${threespeak.id} threespeak ~~~` + ); + + if (links) links.add(threespeak.canonical); + } catch (error) { + console.log(error); + } + + return child; +} + +/** + * Pre-process HTML codes from the Markdown before it gets transformed + * @param child + * @returns {string} + */ +export function preprocessHtml(child) { + try { + if (typeof child === 'string') { + // If typeof child is a string, this means we are trying to process the HTML + // to replace the image/anchor tag created by 3Speak dApp + const threespeak = extractContentId(child); + if (threespeak) { + child = child.replace( + regex.htmlReplacement, + `~~~ embed:${threespeak.fullId} threespeak ~~~` + ); + } + } + } catch (error) { + console.log(error); + } + + return child; +} diff --git a/src/app/components/elements/EmbeddedPlayers/twitch.jsx b/src/app/components/elements/EmbeddedPlayers/twitch.jsx new file mode 100644 index 0000000000000000000000000000000000000000..4c93c61766044cc4c81e6a632de349a189523176 --- /dev/null +++ b/src/app/components/elements/EmbeddedPlayers/twitch.jsx @@ -0,0 +1,114 @@ +import React from 'react'; + +/** + * Regular expressions for detecting and validating provider URLs + * @type {{htmlReplacement: RegExp, main: RegExp, sanitize: RegExp}} + */ +const regex = { + sanitize: /^(https?:)?\/\/player.twitch.tv\/.*/i, + main: /https?:\/\/(?:www.)?twitch.tv\/(?:(videos)\/)?([a-zA-Z0-9][\w]{3,24})/i, +}; + +export default regex; + +/** + * Check if the iframe code in the post editor is to an allowed URL + * <iframe src="https://player.twitch.tv/?channel=tfue" frameborder="0" allowfullscreen="true" scrolling="no" height="378" width="620"></iframe> + * @param url + * @returns {boolean|*} + */ +export function validateIframeUrl(url) { + const match = url.match(regex.sanitize); + + if (match) { + return url; + } + + return false; +} + +/** + * Rewrites the embedded URL to a normalized format + * @param url + * @returns {string|boolean} + */ +export function normalizeEmbedUrl(url) { + const match = url.match(regex.main); + + if (match && match.length >= 3) { + if (match[1] === undefined) { + return `https://player.twitch.tv/?autoplay=false&channel=${ + match[2] + }`; + } + + return `https://player.twitch.tv/?autoplay=false&video=${match[1]}`; + } + + return false; +} + +/** + * Extract the content ID and other metadata from the URL + * @param data + * @returns {null|{id: *, canonical: string, url: *}} + */ +function extractContentId(data) { + if (!data) return null; + + const m = data.match(regex.main); + + if (!m || m.length < 3) return null; + + return { + id: m[1] === `videos` ? `?video=${m[2]}` : `?channel=${m[2]}`, + url: m[0], + canonical: + m[1] === `videos` + ? `https://player.twitch.tv/?video=${m[2]}` + : `https://player.twitch.tv/?channel=${m[2]}`, + }; +} + +export function embedNode(child, links /*images*/) { + try { + const data = child.data; + const twitch = extractContentId(data); + if (!twitch) return child; + + child.data = data.replace( + twitch.url, + `~~~ embed:${twitch.id} twitch ~~~` + ); + + if (links) links.add(twitch.canonical); + } catch (error) { + console.error(error); + } + + return child; +} + +/** + * Generates the Markdown/HTML code to override the detected URL with an iFrame + * @param idx + * @param threespeakId + * @param w + * @param h + * @returns {*} + */ +export function genIframeMd(idx, id, w, h) { + const url = `https://player.twitch.tv/${id}`; + return ( + <div key={`twitch-${id}-${idx}`} className="videoWrapper"> + <iframe + title="Twitch embedded player" + src={url} + width={w} + height={h} + frameBorder="0" + allowFullScreen + /> + </div> + ); +} diff --git a/src/app/components/elements/EmbeddedPlayers/vimeo.jsx b/src/app/components/elements/EmbeddedPlayers/vimeo.jsx new file mode 100644 index 0000000000000000000000000000000000000000..16be8871d7a5e0bfd7633215f11b70af7c7a858e --- /dev/null +++ b/src/app/components/elements/EmbeddedPlayers/vimeo.jsx @@ -0,0 +1,124 @@ +import React from 'react'; + +/** + * Regular expressions for detecting and validating provider URLs + * @type {{htmlReplacement: RegExp, main: RegExp, sanitize: RegExp}} + */ +const regex = { + sanitize: /^(https?:)?\/\/player.vimeo.com\/video\/([0-9]*)/i, + main: /https?:\/\/(?:vimeo.com\/|player.vimeo.com\/video\/)([0-9]+)\/?(#t=((\d+)s?))?\/?/, + contentId: /(?:vimeo.com\/|player.vimeo.com\/video\/)([0-9]+)/, +}; + +export default regex; + +/** + * Check if the iframe code in the post editor is to an allowed URL + * <iframe src="https://player.vimeo.com/video/179213493" width="640" height="360" frameborder="0" webkitallowfullscreen mozallowfullscreen allowfullscreen></iframe> + * @param url + * @returns {boolean|*} + */ +export function validateIframeUrl(url) { + const match = url.match(regex.sanitize); + + if (!match || match.length !== 3) { + return false; + } + + return 'https://player.vimeo.com/video/' + match[2]; +} + +/** + * Rewrites the embedded URL to a normalized format + * @param url + * @returns {string|boolean} + */ +export function normalizeEmbedUrl(url) { + const match = url.match(regex.contentId); + + if (match && match.length >= 2) { + return `https://player.vimeo.com/video/${match[1]}`; + } + + return false; +} + +/** + * Extract the content ID and other metadata from the URL + * @param data + * @returns {null|{id: *, canonical: string, url: *}} + */ +function extractContentId(data) { + if (!data) return null; + const m = data.match(regex.main); + if (!m || m.length < 2) return null; + + const startTime = m.input.match(/t=(\d+)s?/); + + return { + id: m[1], + url: m[0], + startTime: startTime ? startTime[1] : 0, + canonical: `https://player.vimeo.com/video/${m[1]}`, + // thumbnail: requires a callback - http://stackoverflow.com/questions/1361149/get-img-thumbnails-from-vimeo + }; +} + +/** + * Replaces the URL with a custom Markdown for embedded players + * @param child + * @param links + * @returns {*} + */ +export function embedNode(child, links /*images*/) { + try { + const data = child.data; + const vimeo = extractContentId(data); + if (!vimeo) return child; + + const vimeoRegex = new RegExp(`${vimeo.url}(#t=${vimeo.startTime}s?)?`); + if (vimeo.startTime > 0) { + child.data = data.replace( + vimeoRegex, + `~~~ embed:${vimeo.id} vimeo ${vimeo.startTime} ~~~` + ); + } else { + child.data = data.replace( + vimeoRegex, + `~~~ embed:${vimeo.id} vimeo ~~~` + ); + } + + if (links) links.add(vimeo.canonical); + // if(images) images.add(vimeo.thumbnail) // not available + } catch (error) { + console.log(error); + } + return child; +} + +/** + * Generates the Markdown/HTML code to override the detected URL with an iFrame + * @param idx + * @param threespeakId + * @param w + * @param h + * @returns {*} + */ +export function genIframeMd(idx, id, w, h, startTime) { + const url = `https://player.vimeo.com/video/${id}#t=${startTime}s`; + return ( + <div key={`vimeo-${id}-${idx}`} className="videoWrapper"> + <iframe + title="Vimeo embedded player" + src={url} + width={w} + height={h} + frameBorder="0" + webkitallowfullscreen + mozallowfullscreen + allowFullScreen + /> + </div> + ); +} diff --git a/src/app/components/elements/EmbeddedPlayers/youtube.jsx b/src/app/components/elements/EmbeddedPlayers/youtube.jsx new file mode 100644 index 0000000000000000000000000000000000000000..973ba5c49825e3b1895894d6d758c827cf2ef569 --- /dev/null +++ b/src/app/components/elements/EmbeddedPlayers/youtube.jsx @@ -0,0 +1,130 @@ +import React from 'react'; +import YoutubePreview from 'app/components/elements/YoutubePreview'; + +/** + * Regular expressions for detecting and validating provider URLs + * @type {{htmlReplacement: RegExp, main: RegExp, sanitize: RegExp}} + */ +const regex = { + sanitize: /^(https?:)?\/\/www.youtube.com\/embed\/.*/i, + //main: new RegExp(urlSet({ domain: '(?:(?:.*.)?youtube.com|youtu.be)' }), flags), + main: /(?:https?:\/\/)(?:www\.)?(?:(?:youtube.com\/watch\?v=)|(?:youtu.be\/)|(?:youtube.com\/embed\/))([A-Za-z0-9\_\-]+)[^ ]*/i, + contentId: /(?:(?:youtube.com\/watch\?v=)|(?:youtu.be\/)|(?:youtube.com\/embed\/))([A-Za-z0-9\_\-]+)/i, +}; + +export default regex; + +/** + * Check if the iframe code in the post editor is to an allowed URL + * <iframe width="560" height="315" src="https://www.youtube.com/embed/KOnk7Nbqkhs" frameborder="0" allow="accelerometer; autoplay; encrypted-media; gyroscope; picture-in-picture" allowfullscreen></iframe> + * @param url + * @returns {boolean|*} + */ +export function validateIframeUrl(url) { + const match = url.match(regex.sanitize); + + if (match) { + // strip query string (yt: autoplay=1,controls=0,showinfo=0, etc) + return url.replace(/\?.+$/, ''); + } + + return false; +} + +/** + * Rewrites the embedded URL to a normalized format + * @param url + * @returns {string|boolean} + */ +export function normalizeEmbedUrl(url) { + const match = url.match(regex.contentId); + + if (match && match.length >= 2) { + return `https://youtube.com/embed/${match[1]}`; + } + + return false; +} + +/** + * Extract the content ID and other metadata from the URL + * @param data + * @returns {null|{id: *, canonical: string, url: *}} + */ +export function extractContentId(data) { + if (!data) return null; + + const m1 = data.match(regex.main); + const url = m1 ? m1[0] : null; + + if (!url) return null; + + const m2 = url.match(regex.contentId); + const id = m2 && m2.length >= 2 ? m2[1] : null; + + if (!id) return null; + + const startTime = url.match(/t=(\d+)s?/); + + return { + id, + url, + startTime: startTime ? startTime[1] : 0, + thumbnail: 'https://img.youtube.com/vi/' + id + '/0.jpg', + }; +} + +/** + * Replaces the URL with a custom Markdown for embedded players + * @param child + * @param links + * @returns {*} + */ +export function embedNode(child, links, images) { + try { + const yt = extractContentId(child.data); + + if (!yt) return child; + + if (yt.startTime) { + child.data = child.data.replace( + yt.url, + `~~~ embed:${yt.id} youtube ${yt.startTime} ~~~` + ); + } else { + child.data = child.data.replace( + yt.url, + `~~~ embed:${yt.id} youtube ~~~` + ); + } + + if (links) links.add(yt.url); + if (images) images.add(yt.thumbnail); + } catch (error) { + console.log(error); + } + + return child; +} + +/** + * Generates the Markdown/HTML code to override the detected URL with an iFrame + * @param idx + * @param threespeakId + * @param w + * @param h + * @returns {*} + */ +export function genIframeMd(idx, id, w, h, startTime) { + return ( + <YoutubePreview + key={`youtube-${id}-${idx}`} + width={w} + height={h} + youTubeId={id} + startTime={startTime} + frameBorder="0" + allowFullScreen="true" + /> + ); +} diff --git a/src/app/components/elements/ReplyEditor.jsx b/src/app/components/elements/ReplyEditor.jsx index 797b3c38188ffa4754c3019e12079265cf93665b..6a6183488c1e279772612f00d4c546931bc74e26 100644 --- a/src/app/components/elements/ReplyEditor.jsx +++ b/src/app/components/elements/ReplyEditor.jsx @@ -25,6 +25,7 @@ import { fromJS, Set, OrderedSet } from 'immutable'; import Remarkable from 'remarkable'; import Dropzone from 'react-dropzone'; import tt from 'counterpart'; +import { loadUserTemplates, saveUserTemplates } from 'app/utils/UserTemplates'; const remarkable = new Remarkable({ html: true, linkify: false, breaks: true }); @@ -75,6 +76,7 @@ class ReplyEditor extends React.Component { body: PropTypes.string, // initial value defaultPayoutType: PropTypes.string, payoutType: PropTypes.string, + postTemplateName: PropTypes.string, }; static defaultProps = { @@ -191,6 +193,70 @@ class ReplyEditor extends React.Component { const tp = this.props; const np = nextProps; + if ( + typeof nextProps.postTemplateName !== 'undefined' && + nextProps.postTemplateName !== null + ) { + const { formId } = tp; + + if (nextProps.postTemplateName.indexOf('create_') === 0) { + const { username } = tp; + const { body, title, tags } = ns; + const { payoutType, beneficiaries } = np; + const userTemplates = loadUserTemplates(username); + const newTemplateName = nextProps.postTemplateName.replace( + 'create_', + '' + ); + const newTemplate = { + name: nextProps.postTemplateName.replace('create_', ''), + beneficiaries, + payoutType, + markdown: body !== undefined ? body.value : '', + title: title !== undefined ? title.value : '', + tags: tags !== undefined ? tags.value : '', + }; + + let updated = false; + for (let ui = 0; ui < userTemplates.length; ui += 1) { + if (userTemplates[ui].name === newTemplateName) { + userTemplates[ui] = newTemplate; + updated = true; + } + } + + if (updated === false) { + userTemplates.push(newTemplate); + } + + saveUserTemplates(username, userTemplates); + + this.props.setPostTemplateName(formId, null); + } else { + const userTemplates = loadUserTemplates(nextProps.username); + + for (let ti = 0; ti < userTemplates.length; ti += 1) { + const template = userTemplates[ti]; + if (template.name === nextProps.postTemplateName) { + this.state.body.props.onChange(template.markdown); + this.state.title.props.onChange(template.title); + this.state.tags.props.onChange(template.tags); + this.props.setPayoutType( + formId, + template.payoutType + ); + this.props.setBeneficiaries( + formId, + template.beneficiaries + ); + + this.props.setPostTemplateName(formId, null); + break; + } + } + } + } + // Save curent draft to localStorage if ( ts.body.value !== ns.body.value || @@ -1024,6 +1090,12 @@ export default formId => formId, 'beneficiaries', ]); + const postTemplateName = state.user.getIn([ + 'current', + 'post', + formId, + 'postTemplateName', + ]); beneficiaries = beneficiaries ? beneficiaries.toJS() : []; // Post full @@ -1056,6 +1128,7 @@ export default formId => defaultPayoutType, payoutType, beneficiaries, + postTemplateName, initialValues: { title, body, tags }, formId, }; @@ -1081,6 +1154,13 @@ export default formId => value: fromJS(beneficiaries), }) ), + setPostTemplateName: (formId, postTemplateName) => + dispatch( + userActions.set({ + key: ['current', 'post', formId, 'postTemplateName'], + value: postTemplateName, + }) + ), reply: ({ tags, title, diff --git a/src/app/components/elements/SortOrder/index.jsx b/src/app/components/elements/SortOrder/index.jsx index 050cd58edcc8fce79c05e0da12789b937700f26b..8a6f455e3b0dfdf6ab471ebc50fe6bfc6434b60b 100644 --- a/src/app/components/elements/SortOrder/index.jsx +++ b/src/app/components/elements/SortOrder/index.jsx @@ -26,11 +26,11 @@ const SortOrder = ({ topic, sortOrder, horizontal, pathname }) => { { label: tt('main_menu.trending'), value: `/trending${tag}`, - } /* + }, { label: tt('main_menu.hot'), value: `/hot${tag}`, - },*/, + }, ]; if (!topMenu) { diff --git a/src/app/components/modules/PostAdvancedSettings.jsx b/src/app/components/modules/PostAdvancedSettings.jsx index a94e730914bc0fef148b2ab931ef2f6c6fb9d70d..b9ec90aa86d84c87ef174c236efcbeab94463213 100644 --- a/src/app/components/modules/PostAdvancedSettings.jsx +++ b/src/app/components/modules/PostAdvancedSettings.jsx @@ -4,8 +4,11 @@ import reactForm from 'app/utils/ReactForm'; import { SUBMIT_FORM_ID } from 'shared/constants'; import tt from 'counterpart'; import { fromJS } from 'immutable'; -import BeneficiarySelector from 'app/components/cards/BeneficiarySelector'; -import { validateBeneficiaries } from 'app/components/cards/BeneficiarySelector'; +import BeneficiarySelector, { + validateBeneficiaries, +} from 'app/components/cards/BeneficiarySelector'; +import PostTemplateSelector from 'app/components/cards/PostTemplateSelector'; +import { loadUserTemplates, saveUserTemplates } from 'app/utils/UserTemplates'; import * as userActions from 'app/redux/UserReducer'; @@ -16,7 +19,10 @@ class PostAdvancedSettings extends Component { constructor(props) { super(); - this.state = { payoutType: props.initialPayoutType }; + this.state = { + payoutType: props.initialPayoutType, + postTemplateName: null, + }; this.initForm(props); } @@ -43,6 +49,52 @@ class PostAdvancedSettings extends Component { this.setState({ payoutType: event.target.value }); }; + handleTemplateSelected = postTemplateName => { + const { username } = this.props; + const userTemplates = loadUserTemplates(username); + this.setState({ postTemplateName }); + + if (postTemplateName !== null) { + for (let ti = 0; ti < userTemplates.length; ti += 1) { + const template = userTemplates[ti]; + const { beneficiaries } = this.state; + const newBeneficiaries = { + ...beneficiaries, + }; + + if (template.name === postTemplateName) { + if (template.hasOwnProperty('payoutType')) { + this.setState({ payoutType: template.payoutType }); + } + + if (template.hasOwnProperty('beneficiaries')) { + newBeneficiaries.props.value = template.beneficiaries; + this.setState({ beneficiaries: newBeneficiaries }); + } + + break; + } + } + } + }; + + handleDeleteTemplate = (event, postTemplateName) => { + event.preventDefault(); + event.stopPropagation(); + + const { username } = this.props; + const userTemplates = loadUserTemplates(username); + let ui = userTemplates.length; + while (ui--) { + if (userTemplates[ui].name === postTemplateName) { + userTemplates.splice(ui, 1); + } + } + + saveUserTemplates(username, [...userTemplates]); + this.setState({ postTemplateName: null }); + }; + render() { const { formId, @@ -50,10 +102,18 @@ class PostAdvancedSettings extends Component { defaultPayoutType, initialPayoutType, } = this.props; - const { beneficiaries, payoutType } = this.state; + const { beneficiaries, payoutType, postTemplateName } = this.state; + const loadingTemplate = + postTemplateName && postTemplateName.indexOf('create_') === -1; const { submitting, valid, handleSubmit } = this.state.advancedSettings; + const userTemplates = loadUserTemplates(username); const disabled = - submitting || !(valid || payoutType !== initialPayoutType); + submitting || + !( + valid || + payoutType !== initialPayoutType || + postTemplateName !== null + ); const form = ( <form @@ -66,6 +126,10 @@ class PostAdvancedSettings extends Component { if (!err) { this.props.setPayoutType(formId, payoutType); this.props.setBeneficiaries(formId, data.beneficiaries); + this.props.setPostTemplateName( + formId, + postTemplateName + ); this.props.hideAdvancedSettings(); } })} @@ -87,7 +151,7 @@ class PostAdvancedSettings extends Component { <div className="row"> <div className="small-12 medium-6 large-12 columns"> <select - defaultValue={payoutType} + value={payoutType} onChange={this.handlePayoutChange} > <option value="0%"> @@ -129,6 +193,11 @@ class PostAdvancedSettings extends Component { </h4> </div> <BeneficiarySelector {...beneficiaries.props} tabIndex={1} /> + <PostTemplateSelector + username={username} + onChange={this.handleTemplateSelected} + templates={userTemplates} + /> <div className="error"> {(beneficiaries.touched || beneficiaries.value) && beneficiaries.error} @@ -142,8 +211,31 @@ class PostAdvancedSettings extends Component { disabled={disabled} tabIndex={2} > - {tt('g.save')} + {loadingTemplate && + tt( + 'post_advanced_settings_jsx.load_template' + )} + {!loadingTemplate && tt('g.save')} </button> + {loadingTemplate && ( + <button + className="button" + tabIndex={2} + onClick={event => { + this.handleDeleteTemplate( + event, + postTemplateName + ); + }} + > + {postTemplateName && + postTemplateName.indexOf('create_') === + -1 && + tt( + 'post_advanced_settings_jsx.delete_template' + )} + </button> + )} </span> </div> </div> @@ -217,5 +309,14 @@ export default connect( value: fromJS(beneficiaries), }) ), + setPostTemplateName: (formId, postTemplateName, create = false) => + dispatch( + userActions.set({ + key: ['current', 'post', formId, 'postTemplateName'], + value: create + ? `create_${postTemplateName}` + : postTemplateName, + }) + ), }) )(PostAdvancedSettings); diff --git a/src/app/components/modules/Settings.jsx b/src/app/components/modules/Settings.jsx index 3fc2997d5ba9d499ac1ccb281375df7e24e300e4..19c8df907943188cf53c9d9f3d35de7c1ee5dfea 100644 --- a/src/app/components/modules/Settings.jsx +++ b/src/app/components/modules/Settings.jsx @@ -10,6 +10,7 @@ import reactForm from 'app/utils/ReactForm'; import Dropzone from 'react-dropzone'; import MuteList from 'app/components/elements/MuteList'; import { isLoggedIn } from 'app/utils/UserUtil'; +import * as api from '@hiveio/hive-js'; class Settings extends React.Component { constructor(props) { @@ -232,6 +233,36 @@ class Settings extends React.Component { this.props.setUserPreferences(userPreferences); }; + generateAPIEndpointOptions = () => { + let endpoints = api.config.get('alternative_api_endpoints'); + let preferred_api_endpoint = ''; + if (typeof window !== 'undefined') + preferred_api_endpoint = + localStorage.getItem('user_preferred_api_endpoint') === null + ? 'https://api.hive.blog' + : localStorage.getItem('user_preferred_api_endpoint'); + if (endpoints === null || endpoints === undefined) { + return null; + } + let entries = []; + for (var endpoint of endpoints) { + if (endpoint === preferred_api_endpoint) continue; //this one is always present even if the api config call fails + let entry = <option value={endpoint}>{endpoint}</option>; + entries.push(entry); + } + return entries; + }; + + handlePreferredAPIEndpointChange = event => { + if (typeof window !== 'undefined') { + localStorage.setItem( + 'user_preferred_api_endpoint', + event.target.value + ); + api.api.setOptions({ url: event.target.value }); + } + }; + render() { const { state, props } = this; const { @@ -261,6 +292,14 @@ class Settings extends React.Component { progress, } = this.state; + let preferred_api_endpoint = 'https://api.hive.blog'; + if (typeof window !== 'undefined') { + preferred_api_endpoint = + localStorage.getItem('user_preferred_api_endpoint') === null + ? 'https://api.hive.blog' + : localStorage.getItem('user_preferred_api_endpoint'); + } + return ( <div className="Settings"> <div className="row"> @@ -511,6 +550,24 @@ class Settings extends React.Component { </option> </select> </label> + + <label> + {tt( + 'settings_jsx.choose_preferred_api_endpoint' + )} + <select + defaultValue={preferred_api_endpoint} + onChange={ + this.handlePreferredAPIEndpointChange + } + > + <option value={preferred_api_endpoint}> + {preferred_api_endpoint} + </option> + + {this.generateAPIEndpointOptions()} + </select> + </label> <br /> </div> </div> diff --git a/src/app/components/modules/SidePanel/index.jsx b/src/app/components/modules/SidePanel/index.jsx index 2b449b62401396a765e9e3c635b6d43c49acd175..8e6bfef20b3c93c21e620a2e999d4d6fd1199734 100644 --- a/src/app/components/modules/SidePanel/index.jsx +++ b/src/app/components/modules/SidePanel/index.jsx @@ -122,7 +122,7 @@ const SidePanel = ({ organizational: [ { label: tt('navigation.api_docs'), - link: 'https://developers.hive.blog/', + link: 'https://developers.hive.io/', }, { label: tt('navigation.bluepaper'), diff --git a/src/app/locales/en.json b/src/app/locales/en.json index 1e593e74e272d1074806190a6fb35cfcc3a47267..81e6fb01a88ed93ba3b0012adcb3a8b160e0e70c 100644 --- a/src/app/locales/en.json +++ b/src/app/locales/en.json @@ -361,6 +361,15 @@ "beneficiary_percent_total_invalid": "Beneficiary total percentage must be less than 100" }, + "post_template_selector_jsx": { + "templates": "Post templates", + "templates_description": + "You can create post templates locally and load them to create a new blog post. All advanced settings will also be saved and loaded.", + "choose_template": "-- Choose a template to load --", + "create_template_first": "Please create a template first", + "new_template_name": "Name of a new template", + "template_saved": "Template saved successfully" + }, "category_selector_jsx": { "tag_your_story": "Tag (up to 8 tags), the first tag is your main category.", @@ -383,7 +392,9 @@ "payout_option_description": "What type of tokens do you want as rewards from this post?", "current_default": "Default", - "update_default_in_settings": "Update" + "update_default_in_settings": "Update", + "load_template": "Load template", + "delete_template": "Delete template" }, "postfull_jsx": { "this_post_is_not_available_due_to_a_copyright_claim": @@ -712,7 +723,8 @@ "profile_about": "About", "profile_location": "Location", "profile_website": "Website", - "saved": "Saved!" + "saved": "Saved!", + "choose_preferred_api_endpoint": "Choose Your Preferred API Node" }, "transfer_jsx": { "amount_is_in_form": "Amount is in the form 99999.999", diff --git a/src/app/locales/es.json b/src/app/locales/es.json index dac20e857995b4de88cbccb672f3e94e0ae2d008..181d104dc4ec07f5a865a4957879c0a2bb7e19fe 100644 --- a/src/app/locales/es.json +++ b/src/app/locales/es.json @@ -623,7 +623,8 @@ "profile_name": "Nombre de visualización", "profile_about": "Sobre", "profile_location": "Localización", - "profile_website": "Página Web" + "profile_website": "Página Web", + "choose_preferred_api_endpoint": "Elija su nodo de API preferido" }, "transfer_jsx": { "amount_is_in_form": "La cantidad está en el formato 99999.999", diff --git a/src/app/locales/fr.json b/src/app/locales/fr.json index f9f5768df1a425fa04418b6d773bf4f692d2c166..7428497c64be2df02aefc16781aebeddd0f62ea7 100644 --- a/src/app/locales/fr.json +++ b/src/app/locales/fr.json @@ -645,7 +645,8 @@ "profile_name": "Nom à afficher", "profile_about": "A propos", "profile_location": "Lieu", - "profile_website": "Site internet" + "profile_website": "Site internet", + "choose_preferred_api_endpoint": "Choisissez votre nÅ“ud d'API préféré" }, "transfer_jsx": { "amount_is_in_form": "Montant dans le format 99999.999", diff --git a/src/app/locales/it.json b/src/app/locales/it.json index dfb13a0df1a2b254dceba8d187d8daba20faf91e..83dbb5dcf5178826521b141849786cbdbbba885a 100644 --- a/src/app/locales/it.json +++ b/src/app/locales/it.json @@ -630,7 +630,8 @@ "profile_name": "Mostra nome", "profile_about": "About", "profile_location": "Località ", - "profile_website": "Sito" + "profile_website": "Sito", + "choose_preferred_api_endpoint": "Scegli il tuo nodo API preferito" }, "transfer_jsx": { "amount_is_in_form": "La quantita deve essere della forma 99999.999", diff --git a/src/app/locales/ja.json b/src/app/locales/ja.json index 076c3f5140a31269edf3de78a84b7c57519f98fa..697d657b7b0a8e0ec61ac334aa17a76f604dbf59 100644 --- a/src/app/locales/ja.json +++ b/src/app/locales/ja.json @@ -627,7 +627,8 @@ "profile_name": "表示å", "profile_about": "概è¦", "profile_location": "å ´æ‰€", - "profile_website": "ウェブサイト" + "profile_website": "ウェブサイト", + "choose_preferred_api_endpoint": "優先APIãƒŽãƒ¼ãƒ‰ã‚’é¸æŠžã—ã¦ãã ã•ã„" }, "transfer_jsx": { "amount_is_in_form": "金é¡ã¯99999.999ã®å½¢å¼ã§ã™", diff --git a/src/app/locales/ko.json b/src/app/locales/ko.json index d2e4037d6d45057ecae6105727cbdc33a8610f36..5f338219c139b424f763b39bede55ef06e1a1b1f 100644 --- a/src/app/locales/ko.json +++ b/src/app/locales/ko.json @@ -625,7 +625,8 @@ "profile_name": "닉네임", "profile_about": "한 줄 소개", "profile_location": "ì§€ì—", - "profile_website": "웹사ì´íЏ" + "profile_website": "웹사ì´íЏ", + "choose_preferred_api_endpoint": "기본 API 노드 ì„ íƒ" }, "transfer_jsx": { "amount_is_in_form": "Amount is in the form 99999.999", diff --git a/src/app/locales/pl.json b/src/app/locales/pl.json index 926526cc53e739f240f225f83b692ff03021bf12..bcc6b05475f2dc4c8c573659ffe7d853c4f73107 100644 --- a/src/app/locales/pl.json +++ b/src/app/locales/pl.json @@ -724,7 +724,8 @@ "profile_about": "Opis", "profile_location": "Lokalizacja", "profile_website": "Strona www", - "saved": "Zapisano!" + "saved": "Zapisano!", + "choose_preferred_api_endpoint": "Wybierz preferowany wÄ™zeÅ‚ API" }, "transfer_jsx": { "amount_is_in_form": "Podana kwota musi być w formacie 99999.999", diff --git a/src/app/locales/ru.json b/src/app/locales/ru.json index 18ddc273bec9934c326c54f3e6d5b6e9aa3e0ff8..b9de6fdb0979916f66f3d6755c5c65560885cb0d 100644 --- a/src/app/locales/ru.json +++ b/src/app/locales/ru.json @@ -648,7 +648,9 @@ "profile_name": "Отображаемое имÑ", "profile_about": "О Ñебе", "profile_location": "МеÑтоположение", - "profile_website": "Веб-Ñайт" + "profile_website": "Веб-Ñайт", + "choose_preferred_api_endpoint": + "Выберите Ñвой предпочтительный узел API" }, "transfer_jsx": { "amount_is_in_form": "Сумма должна быть в формате 99999.999", diff --git a/src/app/locales/zh.json b/src/app/locales/zh.json index 4403d15b0be646e45ed940b7450461878a0f6051..856ff7b44fa29ecb65a6a5f9e4ed497731be6181 100644 --- a/src/app/locales/zh.json +++ b/src/app/locales/zh.json @@ -574,7 +574,8 @@ "profile_name": "显示åç§°", "profile_about": "关于", "profile_location": "ä½ç½®", - "profile_website": "网站" + "profile_website": "网站", + "choose_preferred_api_endpoint": "选择您的首选API节点" }, "transfer_jsx": { "amount_is_in_form": "金é¢ä»¥99999.999的形å¼å‡ºçް", diff --git a/src/app/utils/BadActorList.js b/src/app/utils/BadActorList.js index e6730008319672894e6148d331d1d9dc7f43f828..86b6676f2071767e9184f4baa8da5517d1442f96 100644 --- a/src/app/utils/BadActorList.js +++ b/src/app/utils/BadActorList.js @@ -180,9 +180,7 @@ blocktraders blocktrades-com blocktrades-info blocktrades-us -blocktrades.com blocktrades.info -blocktrades.us blocktradess blocktradesss blocktradez diff --git a/src/app/utils/ExtractMeta.js b/src/app/utils/ExtractMeta.js index 1c944023607ee889a8bcbd4b246ee7f743c3aea2..c5d169c92cbe0c59a5a1bda0db88078b16dedc44 100644 --- a/src/app/utils/ExtractMeta.js +++ b/src/app/utils/ExtractMeta.js @@ -1,6 +1,8 @@ import { extractBodySummary, extractImageLink } from 'app/utils/ExtractContent'; -import { objAccessor } from 'app/utils/Accessors'; import { makeCanonicalLink } from 'app/utils/CanonicalLinker.js'; +import { proxifyImageUrl } from 'app/utils/ProxifyUrl'; + +const proxify = (url, size) => proxifyImageUrl(url, size).replace(/ /g, '%20'); const site_desc = 'Communities without borders. A social network owned and operated by its users, powered by Hive.'; @@ -51,7 +53,9 @@ function addPostMeta(metas, content, profile) { metas.push({ name: 'og:url', content: localUrl }); metas.push({ name: 'og:image', - content: image || 'https://hive.blog/images/hive-blog-share.png', + content: + proxify(image, '1200x630') || + 'https://hive.blog/images/hive-blog-share.png', }); metas.push({ name: 'og:description', content: desc }); metas.push({ name: 'og:site_name', content: 'Hive' }); @@ -72,7 +76,9 @@ function addPostMeta(metas, content, profile) { metas.push({ name: 'twitter:description', content: desc }); metas.push({ name: 'twitter:image', - content: image || 'https://hive.blog/images/hive-blog-twshare.png', + content: + proxify(image, '1200x630') || + 'https://hive.blog/images/hive-blog-twshare.png', }); } @@ -102,7 +108,6 @@ function addAccountMeta(metas, accountname, profile) { } function readProfile(chain_data, account) { - const profiles = chain_data.profiles; if (!chain_data.profiles[account]) return {}; return chain_data.profiles[account]['metadata']['profile']; } diff --git a/src/app/utils/Links.js b/src/app/utils/Links.js index 164255752f56ca54242b68e11f2e25231929077c..5d75cec64fcf744856cc3045aee8981089550eb7 100644 --- a/src/app/utils/Links.js +++ b/src/app/utils/Links.js @@ -28,8 +28,6 @@ export const remote = (flags = 'i') => urlSet({ domain: `(?!localhost|(?:.*\\.)?steemit.com)${domainPath}` }), flags ); -export const youTube = (flags = 'i') => - new RegExp(urlSet({ domain: '(?:(?:.*.)?youtube.com|youtu.be)' }), flags); export const image = (flags = 'i') => new RegExp(urlSet({ path: imagePath }), flags); export const imageFile = (flags = 'i') => new RegExp(imagePath, flags); @@ -42,17 +40,6 @@ export default { remote: remote(), image: image(), imageFile: imageFile(), - youTube: youTube(), - youTubeId: /(?:(?:youtube.com\/watch\?v=)|(?:youtu.be\/)|(?:youtube.com\/embed\/))([A-Za-z0-9\_\-]+)/i, - vimeo: /https?:\/\/(?:vimeo.com\/|player.vimeo.com\/video\/)([0-9]+)\/?(#t=((\d+)s?))?\/?/, - vimeoId: /(?:vimeo.com\/|player.vimeo.com\/video\/)([0-9]+)/, - // simpleLink: new RegExp(`<a href="(.*)">(.*)<\/a>`, 'ig'), - ipfsPrefix: /(https?:\/\/.*)?\/ipfs/i, - twitch: /https?:\/\/(?:www.)?twitch.tv\/(?:(videos)\/)?([a-zA-Z0-9][\w]{3,24})/i, - dtube: /https:\/\/(?:emb\.)?(?:d.tube\/\#\!\/(?:v\/)?)([a-zA-Z0-9\-\.\/]*)/, - dtubeId: /(?:d\.tube\/#!\/(?:v\/)?([a-zA-Z0-9\-\.\/]*))+/, - threespeak: /(?:https?:\/\/(?:(?:3speak.online\/watch\?v=)|(?:3speak.online\/embed\?v=)))([A-Za-z0-9\_\-\/]+)(&.*)?/i, - threespeakImageLink: /<a href="(https?:\/\/3speak.online\/watch\?v=([A-Za-z0-9\_\-\/]+))".*<img.*?><\/a>/i, }; //TODO: possible this should go somewhere else. diff --git a/src/app/utils/Links.test.js b/src/app/utils/Links.test.js index 4f9d494bdf7bc9ff0b0bbb1b5ad3bca113b436e7..8fea49b49797d4f991d7e8cb3cfb25d729301db8 100644 --- a/src/app/utils/Links.test.js +++ b/src/app/utils/Links.test.js @@ -2,6 +2,8 @@ import assert from 'assert'; import secureRandom from 'secure-random'; import links, * as linksRe from 'app/utils/Links'; import { PARAM_VIEW_MODE, VIEW_MODE_WHISTLE } from '../../shared/constants'; +import youtubeRegex from 'app/components/elements/EmbeddedPlayers/youtube'; +import threespeakRegex from 'app/components/elements/EmbeddedPlayers/threespeak'; describe('Links', () => { it('all', () => { @@ -241,31 +243,31 @@ describe('Performance', () => { } }); it('youTube', () => { - match(linksRe.youTube(), 'https://youtu.be/xG7ajrbj4zs?t=7s'); + match(youtubeRegex.main, 'https://youtu.be/xG7ajrbj4zs?t=7s'); match( - linksRe.youTube(), + youtubeRegex.main, 'https://www.youtube.com/watch?v=xG7ajrbj4zs&t=14s' ); match( - linksRe.youTube(), + youtubeRegex.main, 'https://www.youtube.com/watch?v=xG7ajrbj4zs&feature=youtu.be&t=14s' ); }); it('youTubeId', () => { match( - links.youTubeId, + youtubeRegex.contentId, 'https://youtu.be/xG7ajrbj4zs?t=7s', 'xG7ajrbj4zs', 1 ); match( - links.youTubeId, + youtubeRegex.contentId, 'https://www.youtube.com/watch?v=xG7ajrbj4zs&t=14s', 'xG7ajrbj4zs', 1 ); match( - links.youTubeId, + youtubeRegex.contentId, 'https://www.youtube.com/watch?v=xG7ajrbj4zs&feature=youtu.be&t=14s', 'xG7ajrbj4zs', 1 @@ -273,33 +275,33 @@ describe('Performance', () => { }); it('threespeak', () => { match( - links.threespeak, + threespeakRegex.main, 'https://3speak.online/watch?v=artemislives/tvxkobat' ); match( - links.threespeak, + threespeakRegex.main, 'https://3speak.online/watch?v=artemislives/tvxkobat&jwsource=cl' ); match( - links.threespeak, + threespeakRegex.main, 'https://3speak.online/embed?v=artemislives/tvxkobat' ); }); it('threespeakId', () => { match( - links.threespeak, + threespeakRegex.main, 'https://3speak.online/watch?v=artemislives/tvxkobat', 'artemislives/tvxkobat', 1 ); match( - links.threespeak, + threespeakRegex.main, 'https://3speak.online/watch?v=artemislives/tvxkobat&jwsource=cl', 'artemislives/tvxkobat', 1 ); match( - links.threespeak, + threespeakRegex.main, 'https://3speak.online/embed?v=artemislives/tvxkobat', 'artemislives/tvxkobat', 1 @@ -307,8 +309,8 @@ describe('Performance', () => { }); it('threespeakImageLink', () => { match( - links.threespeakImageLink, - '<a href="https://3speak.online/watch?v=artemislives/tvxkobat" rel="noopener" title="This link will take you away from steemit.com"><img src="https://steemitimages.com/640x0/https://img.3speakcontent.online/tvxkobat/post.png"></a>' + threespeakRegex.htmlReplacement, + '<a href="https://3speak.online/watch?v=artemislives/tvxkobat" rel="noopener" title="This link will take you away from steemit.com" class="steem-keychain-checked"><img src="https://steemitimages.com/640x0/https://img.3speakcontent.online/tvxkobat/post.png"></a>' ); }); }); diff --git a/src/app/utils/SanitizeConfig.js b/src/app/utils/SanitizeConfig.js index 48afa7d88316bd3e2c816c98d466787debf11c55..3d0bc7c54c6f7c910e9573673b5cf5ea3e3233f0 100644 --- a/src/app/utils/SanitizeConfig.js +++ b/src/app/utils/SanitizeConfig.js @@ -8,61 +8,8 @@ import { getExternalLinkWarningMessage, } from 'shared/HtmlReady'; // the only allowable title attributes for div and a tags -const iframeWhitelist = [ - { - re: /^(https?:)?\/\/player.vimeo.com\/video\/.*/i, - fn: src => { - // <iframe src="https://player.vimeo.com/video/179213493" width="640" height="360" frameborder="0" webkitallowfullscreen mozallowfullscreen allowfullscreen></iframe> - if (!src) return null; - const m = src.match( - /https:\/\/player\.vimeo\.com\/video\/([0-9]+)/ - ); - if (!m || m.length !== 2) return null; - return 'https://player.vimeo.com/video/' + m[1]; - }, - }, - { - re: /^(https?:)?\/\/www.youtube.com\/embed\/.*/i, - fn: src => { - return src.replace(/\?.+$/, ''); // strip query string (yt: autoplay=1,controls=0,showinfo=0, etc) - }, - }, - { - re: /^(https?:)?\/\/3speak.online\/embed\?v=.*/i, - fn: src => { - return src; - }, - }, - { - re: /^https:\/\/w.soundcloud.com\/player\/.*/i, - fn: src => { - if (!src) return null; - // <iframe width="100%" height="450" scrolling="no" frameborder="no" src="https://w.soundcloud.com/player/?url=https%3A//api.soundcloud.com/tracks/257659076&auto_play=false&hide_related=false&show_comments=true&show_user=true&show_reposts=false&visual=true"></iframe> - const m = src.match(/url=(.+?)&/); - if (!m || m.length !== 2) return null; - return ( - 'https://w.soundcloud.com/player/?url=' + - m[1] + - '&auto_play=false&hide_related=false&show_comments=true' + - '&show_user=true&show_reposts=false&visual=true' - ); - }, - }, - { - re: /^(https?:)?\/\/player.twitch.tv\/.*/i, - fn: src => { - //<iframe src="https://player.twitch.tv/?channel=ninja" frameborder="0" allowfullscreen="true" scrolling="no" height="378" width="620"> - return src; - }, - }, - { - re: /^https:\/\/emb.d.tube\/\#\!\/([a-zA-Z0-9\-\.\/]+)$/, - fn: src => { - // <iframe width="560" height="315" src="https://emb.d.tube/#!/justineh/u6qoydvy" frameborder="0" allowfullscreen></iframe> - return src; - }, - }, -]; +import { validateIframeUrl as validateEmbbeddedPlayerIframeUrl } from 'app/components/elements/EmbeddedPlayers'; + export const noImageText = '(Image not shown due to low ratings)'; export const allowedTags = ` div, iframe, del, @@ -111,31 +58,29 @@ export default ({ transformTags: { iframe: (tagName, attribs) => { const srcAtty = attribs.src; - for (const item of iframeWhitelist) - if (item.re.test(srcAtty)) { - const src = - typeof item.fn === 'function' - ? item.fn(srcAtty, item.re) - : srcAtty; - if (!src) break; - return { - tagName: 'iframe', - attribs: { - frameborder: '0', - allowfullscreen: 'allowfullscreen', - webkitallowfullscreen: 'webkitallowfullscreen', // deprecated but required for vimeo : https://vimeo.com/forums/help/topic:278181 - mozallowfullscreen: 'mozallowfullscreen', // deprecated but required for vimeo - src, - width: large ? '640' : '480', - height: large ? '360' : '270', - }, - }; - } + const validUrl = validateEmbbeddedPlayerIframeUrl(srcAtty); + + if (validUrl !== false) { + return { + tagName: 'iframe', + attribs: { + frameborder: '0', + allowfullscreen: 'allowfullscreen', + webkitallowfullscreen: 'webkitallowfullscreen', // deprecated but required for vimeo : https://vimeo.com/forums/help/topic:278181 + mozallowfullscreen: 'mozallowfullscreen', // deprecated but required for vimeo + src: validUrl, + width: large ? '640' : '480', + height: large ? '360' : '270', + }, + }; + } + console.log( 'Blocked, did not match iframe "src" white list urls:', tagName, attribs ); + sanitizeErrors.push('Invalid iframe URL: ' + srcAtty); return { tagName: 'div', text: `(Unsupported ${srcAtty})` }; }, diff --git a/src/app/utils/SlateEditor/Iframe.js b/src/app/utils/SlateEditor/Iframe.jsx similarity index 66% rename from src/app/utils/SlateEditor/Iframe.js rename to src/app/utils/SlateEditor/Iframe.jsx index fd2bc705557fd31a39710c0992e2bbcc07744751..147f01c688ce5239222b8cdcc71cad8e718e279d 100644 --- a/src/app/utils/SlateEditor/Iframe.js +++ b/src/app/utils/SlateEditor/Iframe.jsx @@ -1,47 +1,11 @@ import React from 'react'; -import linksRe from 'app/utils/Links'; +import { normalizeEmbedUrl as normalizeEmbbeddedPlayerEmbedUrl } from 'app/components/elements/EmbeddedPlayers'; export default class Iframe extends React.Component { normalizeEmbedUrl = url => { - let match; - - // Detect youtube URLs - match = url.match(linksRe.youTubeId); - if (match && match.length >= 2) { - return 'https://www.youtube.com/embed/' + match[1]; - } - - // Detect vimeo - match = url.match(linksRe.vimeoId); - if (match && match.length >= 2) { - return 'https://player.vimeo.com/video/' + match[1]; - } - - // Detect twitch stream - match = url.match(linksRe.twitch); - if (match && match.length >= 3) { - if (match[1] === undefined) { - return ( - 'https://player.twitch.tv/?autoplay=false&channel=' + - match[2] - ); - } else { - return ( - 'https://player.twitch.tv/?autoplay=false&video=' + match[1] - ); - } - } - - // Detect dtube - match = url.match(linksRe.dtubeId); - if (match && match.length >= 2) { - return 'https://emb.d.tube/#!/' + match[1]; - } - - // Detect 3Speak - match = url.match(linksRe.threespeak); - if (match && match.length >= 2) { - return 'https://3speak.online/embed?v=' + match[1]; + const validEmbedUrl = normalizeEmbbeddedPlayerEmbedUrl(url); + if (validEmbedUrl !== false) { + return validEmbedUrl; } console.log('unable to auto-detect embed url', url); @@ -49,7 +13,7 @@ export default class Iframe extends React.Component { }; onChange = e => { - const { node, state, editor } = this.props; + const { node, editor } = this.props; const value = e.target.value; const src = this.normalizeEmbedUrl(value) || value; diff --git a/src/app/utils/UserTemplates.js b/src/app/utils/UserTemplates.js new file mode 100644 index 0000000000000000000000000000000000000000..99e99218f3bb37559f88208dc05c0edbec737a63 --- /dev/null +++ b/src/app/utils/UserTemplates.js @@ -0,0 +1,16 @@ +export const loadUserTemplates = username => { + const lsEntryName = `hivePostTemplates-${username}`; + let userTemplates = window.localStorage.getItem(lsEntryName); + if (userTemplates) { + userTemplates = JSON.parse(userTemplates); + } else { + userTemplates = []; + } + + return userTemplates; +}; + +export const saveUserTemplates = (username, templates) => { + const lsEntryName = `hivePostTemplates-${username}`; + window.localStorage.setItem(lsEntryName, JSON.stringify(templates)); +}; diff --git a/src/server/index.js b/src/server/index.js index 3ccf0259731e1b79d41491acc01526e88e921b26..41cd23467a351839e1566f7881f1a76b7e234118 100644 --- a/src/server/index.js +++ b/src/server/index.js @@ -30,6 +30,8 @@ global.$STM_Config = { site_domain: config.get('site_domain'), google_analytics_id: config.get('google_analytics_id'), wallet_url: config.get('wallet_url'), + failover_threshold: config.get('failover_threshold'), + alternative_api_endpoints: config.get('alternative_api_endpoints'), }; const WebpackIsomorphicTools = require('webpack-isomorphic-tools'); @@ -50,9 +52,16 @@ global.webpackIsomorphicTools.server(ROOT, () => { randomize: true, }, useAppbaseApi: !!config.steemd_use_appbase, + alternative_api_endpoints: config.get('alternative_api_endpoints'), + failover_threshold: config.get('failover_threshold'), }); steem.config.set('address_prefix', config.get('address_prefix')); steem.config.set('chain_id', config.get('chain_id')); + steem.config.set( + 'alternative_api_endpoints', + config.get('alternative_api_endpoints') + ); + steem.config.set('failover_threshold', config.get('failover_threshold')); // const CliWalletClient = require('shared/api_client/CliWalletClient').default; // if (process.env.NODE_ENV === 'production') connect_promises.push(CliWalletClient.instance().connect_promise()); diff --git a/src/shared/HtmlReady.js b/src/shared/HtmlReady.js index 1436f654df601aec83f9a71be5748916bd80a070..b735e0795c011f0bed6b11a4e33bd6634bffdfc2 100644 --- a/src/shared/HtmlReady.js +++ b/src/shared/HtmlReady.js @@ -4,6 +4,11 @@ import linksRe, { any as linksAny } from 'app/utils/Links'; import { validate_account_name } from 'app/utils/ChainValidation'; import { proxifyImageUrl } from 'app/utils/ProxifyUrl'; import * as Phishing from 'app/utils/Phishing'; +import { + embedNode as EmbeddedPlayerEmbedNode, + preprocessHtml, +} from 'app/components/elements/EmbeddedPlayers'; +import { extractContentId as youTubeId } from 'app/components/elements/EmbeddedPlayers/youtube'; export const getPhishingWarningMessage = () => tt('g.phishy_message'); export const getExternalLinkWarningMessage = () => @@ -124,13 +129,6 @@ export default function(html, { mutate = true, hideImages = false } = {}) { } } -function preprocessHtml(html) { - // Replacing 3Speak Image/Anchor tag with an embedded player - html = embedThreeSpeakNode(html); - - return html; -} - function traverse(node, state, depth = 0) { if (!node || !node.childNodes) return; Array(...node.childNodes).forEach(child => { @@ -181,6 +179,8 @@ function link(state, child) { // wrap iframes in div.videoWrapper to control size/aspect ratio function iframe(state, child) { const url = child.getAttribute('src'); + + // @TODO move this into the centralized EmbeddedPlayer code if (url) { const { images, links } = state; const yt = youTubeId(url); @@ -245,11 +245,8 @@ function linkifyNode(child, state) { const { mutate } = state; if (!child.data) return; - child = embedYouTubeNode(child, state.links, state.images); - child = embedVimeoNode(child, state.links, state.images); - child = embedTwitchNode(child, state.links, state.images); - child = embedDTubeNode(child, state.links, state.images); - child = embedThreeSpeakNode(child, state.links, state.images); + + child = EmbeddedPlayerEmbedNode(child, state.links, state.images); const data = XMLSerializer.serializeToString(child); const content = linkify( @@ -325,206 +322,6 @@ function linkify(content, mutate, hashtags, usertags, images, links) { return content; } -function embedYouTubeNode(child, links, images) { - try { - const data = child.data; - const yt = youTubeId(data); - if (!yt) return child; - - if (yt.startTime) { - child.data = data.replace( - yt.url, - `~~~ embed:${yt.id} youtube ${yt.startTime} ~~~` - ); - } else { - child.data = data.replace(yt.url, `~~~ embed:${yt.id} youtube ~~~`); - } - - if (links) links.add(yt.url); - if (images) images.add(yt.thumbnail); - } catch (error) { - console.error('yt_node', error); - } - return child; -} - -/** @return {id, url} or <b>null</b> */ -function youTubeId(data) { - if (!data) return null; - - const m1 = data.match(linksRe.youTube); - const url = m1 ? m1[0] : null; - if (!url) return null; - - const m2 = url.match(linksRe.youTubeId); - const id = m2 && m2.length >= 2 ? m2[1] : null; - if (!id) return null; - - const startTime = url.match(/t=(\d+)s?/); - - return { - id, - url, - startTime: startTime ? startTime[1] : 0, - thumbnail: 'https://img.youtube.com/vi/' + id + '/0.jpg', - }; -} - -/** @return {id, url} or <b>null</b> */ -function getThreeSpeakId(data) { - if (!data) return null; - - const match = data.match(linksRe.threespeak); - const url = match ? match[0] : null; - if (!url) return null; - const fullId = match[1]; - const id = fullId.split('/').pop(); - - return { - id, - fullId, - url, - thumbnail: `https://img.3speakcontent.online/${id}/post.png`, - }; -} - -function embedThreeSpeakNode(child, links, images) { - try { - if (typeof child === 'string') { - // If typeof child is a string, this means we are trying to process the HTML - // to replace the image/anchor tag created by 3Speak dApp - const threespeakId = getThreeSpeakId(child); - if (threespeakId) { - child = child.replace( - linksRe.threespeakImageLink, - `~~~ embed:${threespeakId.fullId} threespeak ~~~` - ); - } - } else { - // If child is not a string, we are processing plain text - // to replace a bare URL - const data = child.data; - const threespeakId = getThreeSpeakId(data); - if (!threespeakId) return child; - - child.data = data.replace( - threespeakId.url, - `~~~ embed:${threespeakId.fullId} threespeak ~~~` - ); - - if (links) links.add(threespeakId.url); - if (images) images.add(threespeakId.thumbnail); - } - } catch (error) { - console.log(error); - } - - return child; -} - -function embedVimeoNode(child, links /*images*/) { - try { - const data = child.data; - const vimeo = vimeoId(data); - if (!vimeo) return child; - - const vimeoRegex = new RegExp(`${vimeo.url}(#t=${vimeo.startTime}s?)?`); - if (vimeo.startTime > 0) { - child.data = data.replace( - vimeoRegex, - `~~~ embed:${vimeo.id} vimeo ${vimeo.startTime} ~~~` - ); - } else { - child.data = data.replace( - vimeoRegex, - `~~~ embed:${vimeo.id} vimeo ~~~` - ); - } - - if (links) links.add(vimeo.canonical); - // if(images) images.add(vimeo.thumbnail) // not available - } catch (error) { - console.error('vimeo_embed', error); - } - return child; -} - -function vimeoId(data) { - if (!data) return null; - const m = data.match(linksRe.vimeo); - if (!m || m.length < 2) return null; - - const startTime = m.input.match(/t=(\d+)s?/); - - return { - id: m[1], - url: m[0], - startTime: startTime ? startTime[1] : 0, - canonical: `https://player.vimeo.com/video/${m[1]}`, - // thumbnail: requires a callback - http://stackoverflow.com/questions/1361149/get-img-thumbnails-from-vimeo - }; -} - -function embedTwitchNode(child, links /*images*/) { - try { - const data = child.data; - const twitch = twitchId(data); - if (!twitch) return child; - - child.data = data.replace( - twitch.url, - `~~~ embed:${twitch.id} twitch ~~~` - ); - - if (links) links.add(twitch.canonical); - } catch (error) { - console.error('twitch_error', error); - } - return child; -} - -function twitchId(data) { - if (!data) return null; - const m = data.match(linksRe.twitch); - if (!m || m.length < 3) return null; - - return { - id: m[1] === `videos` ? `?video=${m[2]}` : `?channel=${m[2]}`, - url: m[0], - canonical: - m[1] === `videos` - ? `https://player.twitch.tv/?video=${m[2]}` - : `https://player.twitch.tv/?channel=${m[2]}`, - }; -} - -function embedDTubeNode(child, links /*images*/) { - try { - const data = child.data; - const dtube = dtubeId(data); - if (!dtube) return child; - - child.data = data.replace(dtube.url, `~~~ embed:${dtube.id} dtube ~~~`); - - if (links) links.add(dtube.canonical); - } catch (error) { - console.error('dtube_embed', error); - } - return child; -} - -function dtubeId(data) { - if (!data) return null; - const m = data.match(linksRe.dtube); - if (!m || m.length < 2) return null; - - return { - id: m[1], - url: m[0], - canonical: `https://emb.d.tube/#!/${m[1]}`, - }; -} - function ipfsPrefix(url) { if ($STM_Config.ipfs_prefix) { // Convert //ipfs/xxx or /ipfs/xxx into https://steemit.com/ipfs/xxxxx diff --git a/src/shared/HtmlReady.test.js b/src/shared/HtmlReady.test.js index 71da0dc140fcbf8aa0d32d354b8c35f8007e5b29..95722e0b44817bb734c15a0ad37bb06905c8e2c3 100644 --- a/src/shared/HtmlReady.test.js +++ b/src/shared/HtmlReady.test.js @@ -210,7 +210,7 @@ describe('htmlready', () => { expect(res).toEqual(htmlified); }); - it('should handle short youtube link start time', () => { + it('should handle short youtu.be link start time', () => { const testString = '<html><p>https://youtu.be/ToQfmnj7FR8?t=4572s</p></html>'; const htmlified = diff --git a/yarn.lock b/yarn.lock index 0e62b0c19ea97483af2d2e7026ceaf4ffae1da07..3807eaef7a4ec713d36f06014a60834c2919d897 100644 --- a/yarn.lock +++ b/yarn.lock @@ -15,10 +15,10 @@ resolved "https://registry.yarnpkg.com/@hivechain/hivescript/-/hivescript-1.1.7.tgz#24d89ef8b0984c6e496934b626a72a374618338a" integrity sha512-hS/negE2DoPqPYzY14+WLqNrdV+4R9pt3AhKkabW6m8auNXOp6i48+ckTMRWT5/lU4/avhjKgOgdqvTAMbXRAg== -"@hiveio/hive-js@0.0.1": - version "0.0.1" - resolved "https://registry.yarnpkg.com/@hiveio/hive-js/-/hive-js-0.0.1.tgz#2acdce5550adc3f2cc23cb3aad17539360770549" - integrity sha512-aFX58gzLzZvZnPJdEafGRkVYNwQY6Lfqqr5yjr/yFNRXahPd2+m/f/n0e84R894DXB1HcgYASsxR/VwOpNscVg== +"@hiveio/hive-js@^0.0.2": + version "0.0.2" + resolved "https://registry.yarnpkg.com/@hiveio/hive-js/-/hive-js-0.0.2.tgz#6f4a6fa133a1528534d410ad5f9bd4fee2b738aa" + integrity sha512-eHNAuYynOTXLTNBpftKEu3raWc2Ot6kj5Fti2OkQdSZaFFryl0hwjI8pH8fTmFnOg7YULOFonYs1Zo+4qhOMeQ== dependencies: "@steemit/rpc-auth" "^1.1.1" bigi "^1.4.2"