├── README.md ├── frontend ├── .env ├── .gitignore ├── .vscode │ └── settings.json ├── GNUmakefile ├── README.md ├── build │ ├── asset-manifest.json │ ├── favicon.ico │ ├── index.html │ ├── manifest.json │ ├── precache-manifest.7f0a1849234fb16afd74cd290c1ec9fc.js │ ├── service-worker.js │ └── static │ │ ├── css │ │ ├── main.54b516df.chunk.css │ │ └── main.54b516df.chunk.css.map │ │ ├── js │ │ ├── 2.8a0cc92f.chunk.js │ │ ├── 2.8a0cc92f.chunk.js.map │ │ ├── main.22e40df3.chunk.js │ │ ├── main.22e40df3.chunk.js.map │ │ ├── runtime-main.9c63dd59.js │ │ └── runtime-main.9c63dd59.js.map │ │ └── media │ │ ├── bg.03b4fd11.jpg │ │ ├── ic_arrow-down-circle.3d1db861.svg │ │ ├── ig_coin_card.968dcb99.png │ │ ├── ig_lose_coin.91f34167.png │ │ ├── ig_win_coin.44207f94.png │ │ └── tt-usdt.6ef85566.png ├── environment.yml ├── package.json ├── public │ ├── favicon.ico │ ├── index.html │ └── manifest.json ├── src │ ├── ContractEnum.ts │ ├── assets │ │ ├── bg.jpg │ │ ├── ic_arrow-down-circle.svg │ │ ├── icon_TT.png │ │ ├── ig_bet_btn.png │ │ ├── ig_coin_card.png │ │ ├── ig_double_btn.png │ │ ├── ig_lose_coin.png │ │ ├── ig_reset_btn.png │ │ ├── ig_win_coin.png │ │ └── tt-usdt.png │ ├── components │ │ ├── App │ │ │ ├── App.css │ │ │ ├── App.tsx │ │ │ ├── DoubleOrNothing.json │ │ │ └── ERC677.json │ │ ├── Coin │ │ │ ├── Coin.css │ │ │ └── Coin.tsx │ │ ├── CoinPicker │ │ │ ├── CoinPicker.css │ │ │ └── CoinPicker.tsx │ │ ├── ContractLoader │ │ │ └── ContractLoader.tsx │ │ ├── Display │ │ │ ├── Display.css │ │ │ └── Display.tsx │ │ ├── Game │ │ │ ├── Game.css │ │ │ └── Game.tsx │ │ ├── Header │ │ │ ├── Header.css │ │ │ └── Header.tsx │ │ ├── ReferralRules │ │ │ ├── ReferralRules.css │ │ │ └── ReferralRules.tsx │ │ └── Web3Loader │ │ │ └── Web3Loader.tsx │ ├── config.ts │ ├── containers │ │ ├── DoubleGameContainer │ │ │ └── DoubleGameContainer.tsx │ │ ├── Erc677GameContainer │ │ │ └── Erc677GameContainer.tsx │ │ └── GameContainer │ │ │ └── GameContainer.tsx │ ├── contexts │ │ ├── ContractContext │ │ │ └── ContractContext.tsx │ │ └── Web3Context │ │ │ └── Web3Context.tsx │ ├── index.css │ ├── index.tsx │ ├── react-app-env.d.ts │ └── serviceWorker.ts ├── tsconfig.json ├── typings │ ├── abi.d.ts │ ├── env.d.ts │ ├── images.d.ts │ ├── sounds.d.ts │ └── web3.d.ts └── yarn.lock ├── index.html └── smart-contracts ├── .gitignore ├── GNUmakefile ├── build └── contracts │ ├── DoubleOrNothing.json │ ├── ERC20.json │ ├── ERC677.json │ ├── ERC677Receiver.json │ ├── IBurnableMintableERC677Token.json │ ├── IERC20.json │ ├── Migrations.json │ ├── Ownable.json │ ├── Referral.json │ ├── SafeMath.json │ └── ThunderRandomLibraryInterface.json ├── contracts ├── DoubleOrNothing.sol ├── ERC677Interface.sol ├── Migrations.sol └── ThunderRandomLibraryInterface.sol ├── environment.yml ├── migrations └── 1_initial_migration.js ├── package.json ├── truffle-config.js └── yarn.lock /README.md: -------------------------------------------------------------------------------- 1 | THIS IS NOT PRODUCTION READY AND SHOULD NOT BE USED IN ANY WAY IN PRODUCTION. 2 | 3 | [DoubleOrNothing](https://thundercore.github.io/DoubleOrNothing/) is a coin flip game built on Thunder. 4 | 5 | # Getting Started 6 | 7 | 1. Fork this repo 8 | 2. Rename the forked repo as {{Your-github-username}}.github.io 9 | 3. ssh `git clone git@github.com:{{Your-github-username}}/{{Your-github-username}}.github.io.git` 10 | \ 11 | or 12 | \ 13 | https `git clone https://github.com/{{Your-github-username}}/{{Your-github-username}}.github.io.git` 14 | 4. `cd {{Your-github-username}}.github.io.git` 15 | 5. `git commit --allow-empty -m "bump"` 16 | 6. `git push` 17 | *Note: If you see the error “push declined due to email privacy restrictions”, you will need to allow command line pushes that expose your email address. This under “Settings”, “Emails” 18 | 7. It will take ~10 mins for Github to create your page. It can be accessed at 19 | \ 20 | `https://{{Your-github-username}}.github.io` 21 | 8. While you wait, make sure to have [Metamask](https://metamask.io/) installed 22 | 9. Set it up to a new Custom RPC url: https://mainnet-rpc.thundercore.com 23 | 10. When your page is built, you should be able to play the game 24 | 25 | 26 | # Understanding the Repo for development 27 | 28 | ### Setup 29 | We will be using node v8 or node v10 30 | - `cd smart-contracts` 31 | - `yarn install` or ` npm install` 32 | - `cd ../frontend` 33 | - `yarn install` or `npm install` 34 | - `yarn start` or `npm run start` 35 | 36 | ### Smart Contracts 37 | 38 | This project uses [Truffle](https://www.trufflesuite.com) to handle all of the smart contract related work. 39 | 40 | In the `/smart-contracts/contracts` folder, you will find `DoubleOrNothing.sol` which contains the game logic. 41 | 42 | After you make any updates to it, you will have to redeploy the contract. 43 | 44 | `yarn migrate --network development/thunder --reset` or `npm run migrate -- --network development/thunder --reset` 45 | 46 | ** Development points to localhost:8545 [Ganache](https://www.trufflesuite.com/ganache) 47 | 48 | ** To deploy on thunder, you must add a mnemonic to `/smart-contracts/truffle-config.js` 49 | 50 | ### Frontend 51 | The frontend is built using the [create-react-app](https://github.com/facebook/create-react-app) project with [Typescript](https://www.typescriptlang.org/). 52 | To interact with the chain, we use [Ethers.js](https://docs.ethers.io/ethers.js/html/). 53 | -------------------------------------------------------------------------------- /frontend/.env: -------------------------------------------------------------------------------- 1 | #REACT_APP_DOUBLE_CONTRACT_ADDRESS=0x48caD7fe30Fd1CA52EcBda9C82E398B0FFd321eE 2 | #REACT_APP_TT_DAI_CONTRACT=0xd60db41a718a73da844a4c454c8bd6e07173d722 3 | #REACT_APP_TT_USDT_CONTRACT=0x42532085e51e5618575005f626589ff57d280d68 4 | #REACT_APP_RPC_URL=https://testnet-rpc.thundercore.com 5 | 6 | REACT_APP_DOUBLE_CONTRACT_ADDRESS=0x8693a164ebc4429fCFF4461B481FCD4E60690fD3 7 | #REACT_APP_TT_DAI_CONTRACT=0x2b31e3b88847f03c1335E99A0d1274A2c72059DE 8 | REACT_APP_TT_USDT_CONTRACT=0x4f3C8E20942461e2c3Bdd8311AC57B0c222f2b82 9 | REACT_APP_RPC_URL=https://mainnet-rpc.thundercore.com 10 | #REACT_APP_RPC_URL=http://localhost:8545 11 | -------------------------------------------------------------------------------- /frontend/.gitignore: -------------------------------------------------------------------------------- 1 | # See https://help.github.com/articles/ignoring-files/ for more about ignoring files. 2 | 3 | # dependencies 4 | /node_modules 5 | /.pnp 6 | .pnp.js 7 | 8 | # testing 9 | /coverage 10 | 11 | # misc 12 | .DS_Store 13 | .env.local 14 | .env.development.local 15 | .env.test.local 16 | .env.production.local 17 | 18 | npm-debug.log* 19 | yarn-debug.log* 20 | yarn-error.log* 21 | 22 | conda-env 23 | 24 | # VIM 25 | *.swp 26 | *.swo 27 | -------------------------------------------------------------------------------- /frontend/.vscode/settings.json: -------------------------------------------------------------------------------- 1 | { 2 | "cSpell.words": [ 3 | "Numberify", 4 | "Usdt" 5 | ] 6 | } -------------------------------------------------------------------------------- /frontend/GNUmakefile: -------------------------------------------------------------------------------- 1 | CONDA := $(HOME)/miniconda3/bin/conda 2 | 3 | TOP_DIR := $(patsubst %/,%,$(dir $(realpath $(lastword $(MAKEFILE_LIST))))) 4 | CONDA_ENV_DIR := $(TOP_DIR)/conda-env 5 | 6 | # This can be used to customized local builds. 7 | -include $(TOP_DIR)/Local.mk 8 | 9 | PATH := $(CONDA_ENV_DIR)/bin:$(TOP_DIR)/node_modules/.bin:$(PATH) 10 | export PATH 11 | 12 | PREFERRED_INTERACTIVE_SHELL ?= bash 13 | PS1_NAME ?= 'double-or-nothing' 14 | MAKE_SHELL_PS1 ?= '$(PS1_NAME) $$ ' 15 | .PHONY: shell 16 | ifeq ($(PREFERRED_INTERACTIVE_SHELL),bash) 17 | shell: 18 | @INIT_FILE=$(shell mktemp); \ 19 | printf '[ -e $$HOME/.bashrc ] && source $$HOME/.bashrc\n' > $$INIT_FILE; \ 20 | printf '[ -e Local.env ] && source Local.env\n' >> $$INIT_FILE; \ 21 | printf 'PS1='"$(MAKE_SHELL_PS1) "'\n' >> $$INIT_FILE; \ 22 | $(PREFERRED_INTERACTIVE_SHELL) --init-file $$INIT_FILE || true 23 | else ifeq ($(PREFERRED_INTERACTIVE_SHELL),fish) 24 | shell: 25 | @INIT_FILE=$(shell mktemp); \ 26 | printf 'if functions -q fish_right_prompt\n' > $$INIT_FILE; \ 27 | printf ' functions -c fish_right_prompt __fish_right_prompt_original\n' >> $$INIT_FILE; \ 28 | printf ' functions -e fish_right_prompt\n' >> $$INIT_FILE; \ 29 | printf 'else\n' >> $$INIT_FILE; \ 30 | printf ' function __fish_right_prompt_original\n' >> $$INIT_FILE; \ 31 | printf ' end\n' >> $$INIT_FILE; \ 32 | printf 'end\n' >> $$INIT_FILE; \ 33 | printf 'function fish_right_prompt\n' >> $$INIT_FILE; \ 34 | printf ' echo -n "($(PS1_NAME)) "\n' >> $$INIT_FILE; \ 35 | printf ' __fish_right_prompt_original\n' >> $$INIT_FILE; \ 36 | printf 'end\n' >> $$INIT_FILE; \ 37 | $(PREFERRED_INTERACTIVE_SHELL) --init-command="source $$INIT_FILE" || true 38 | else 39 | shell: 40 | @$(PREFERRED_INTERACTIVE_SHELL) || true 41 | endif 42 | 43 | .PHONY: conda-env 44 | conda-env: 45 | if [ ! -d $(CONDA_ENV_DIR) ]; then \ 46 | $(CONDA) env create -f environment.yml -p $(CONDA_ENV_DIR); \ 47 | else \ 48 | $(CONDA) env update -f environment.yml -p $(CONDA_ENV_DIR); \ 49 | fi 50 | yarn install 51 | -------------------------------------------------------------------------------- /frontend/README.md: -------------------------------------------------------------------------------- 1 | This project was bootstrapped with [Create React App](https://github.com/facebook/create-react-app). 2 | 3 | ## Available Scripts 4 | 5 | In the project directory, you can run: 6 | 7 | ### `npm start` 8 | 9 | Runs the app in the development mode.
10 | Open [http://localhost:3000](http://localhost:3000) to view it in the browser. 11 | 12 | The page will reload if you make edits.
13 | You will also see any lint errors in the console. 14 | 15 | ### `npm test` 16 | 17 | Launches the test runner in the interactive watch mode.
18 | See the section about [running tests](https://facebook.github.io/create-react-app/docs/running-tests) for more information. 19 | 20 | ### `npm run build` 21 | 22 | Builds the app for production to the `build` folder.
23 | It correctly bundles React in production mode and optimizes the build for the best performance. 24 | 25 | The build is minified and the filenames include the hashes.
26 | Your app is ready to be deployed! 27 | 28 | See the section about [deployment](https://facebook.github.io/create-react-app/docs/deployment) for more information. 29 | 30 | ### `npm run eject` 31 | 32 | **Note: this is a one-way operation. Once you `eject`, you can’t go back!** 33 | 34 | If you aren’t satisfied with the build tool and configuration choices, you can `eject` at any time. This command will remove the single build dependency from your project. 35 | 36 | Instead, it will copy all the configuration files and the transitive dependencies (Webpack, Babel, ESLint, etc) right into your project so you have full control over them. All of the commands except `eject` will still work, but they will point to the copied scripts so you can tweak them. At this point you’re on your own. 37 | 38 | You don’t have to ever use `eject`. The curated feature set is suitable for small and middle deployments, and you shouldn’t feel obligated to use this feature. However we understand that this tool wouldn’t be useful if you couldn’t customize it when you are ready for it. 39 | 40 | ## Learn More 41 | 42 | You can learn more in the [Create React App documentation](https://facebook.github.io/create-react-app/docs/getting-started). 43 | 44 | To learn React, check out the [React documentation](https://reactjs.org/). 45 | -------------------------------------------------------------------------------- /frontend/build/asset-manifest.json: -------------------------------------------------------------------------------- 1 | { 2 | "files": { 3 | "main.css": "/frontend/build/static/css/main.54b516df.chunk.css", 4 | "main.js": "/frontend/build/static/js/main.22e40df3.chunk.js", 5 | "main.js.map": "/frontend/build/static/js/main.22e40df3.chunk.js.map", 6 | "runtime-main.js": "/frontend/build/static/js/runtime-main.9c63dd59.js", 7 | "runtime-main.js.map": "/frontend/build/static/js/runtime-main.9c63dd59.js.map", 8 | "static/js/2.8a0cc92f.chunk.js": "/frontend/build/static/js/2.8a0cc92f.chunk.js", 9 | "static/js/2.8a0cc92f.chunk.js.map": "/frontend/build/static/js/2.8a0cc92f.chunk.js.map", 10 | "index.html": "/frontend/build/index.html", 11 | "precache-manifest.7f0a1849234fb16afd74cd290c1ec9fc.js": "/frontend/build/precache-manifest.7f0a1849234fb16afd74cd290c1ec9fc.js", 12 | "service-worker.js": "/frontend/build/service-worker.js", 13 | "static/css/main.54b516df.chunk.css.map": "/frontend/build/static/css/main.54b516df.chunk.css.map", 14 | "static/media/App.css": "/frontend/build/static/media/bg.03b4fd11.jpg", 15 | "static/media/ic_arrow-down-circle.3d1db861.svg": "/frontend/build/static/media/ic_arrow-down-circle.3d1db861.svg", 16 | "static/media/ig_coin_card.png": "/frontend/build/static/media/ig_coin_card.968dcb99.png", 17 | "static/media/Coin.css": "/frontend/build/static/media/ig_win_coin.44207f94.png", 18 | "static/media/tt-usdt.png": "/frontend/build/static/media/tt-usdt.6ef85566.png" 19 | } 20 | } -------------------------------------------------------------------------------- /frontend/build/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thundercore/DoubleOrNothing/dd0501343022f2c4f5217c42a69fad2280306e84/frontend/build/favicon.ico -------------------------------------------------------------------------------- /frontend/build/index.html: -------------------------------------------------------------------------------- 1 | DoubleOrNothing
-------------------------------------------------------------------------------- /frontend/build/manifest.json: -------------------------------------------------------------------------------- 1 | { 2 | "short_name": "DON", 3 | "name": "Double Or Nothing", 4 | "icons": [ 5 | { 6 | "src": "favicon.ico", 7 | "sizes": "64x64 32x32 24x24 16x16", 8 | "type": "image/x-icon" 9 | } 10 | ], 11 | "start_url": ".", 12 | "display": "standalone", 13 | "theme_color": "#000000", 14 | "background_color": "#ffffff" 15 | } 16 | -------------------------------------------------------------------------------- /frontend/build/precache-manifest.7f0a1849234fb16afd74cd290c1ec9fc.js: -------------------------------------------------------------------------------- 1 | self.__precacheManifest = (self.__precacheManifest || []).concat([ 2 | { 3 | "revision": "0cf4e84343a789d7b726ed63e73c5b45", 4 | "url": "/frontend/build/index.html" 5 | }, 6 | { 7 | "revision": "637e1350797521a4f18d", 8 | "url": "/frontend/build/static/css/main.54b516df.chunk.css" 9 | }, 10 | { 11 | "revision": "61f1d6268fca7f171314", 12 | "url": "/frontend/build/static/js/2.8a0cc92f.chunk.js" 13 | }, 14 | { 15 | "revision": "637e1350797521a4f18d", 16 | "url": "/frontend/build/static/js/main.22e40df3.chunk.js" 17 | }, 18 | { 19 | "revision": "161c561bd634b38eb726", 20 | "url": "/frontend/build/static/js/runtime-main.9c63dd59.js" 21 | }, 22 | { 23 | "revision": "03b4fd11fa86248c71b3aba06a289206", 24 | "url": "/frontend/build/static/media/bg.03b4fd11.jpg" 25 | }, 26 | { 27 | "revision": "3d1db8614cf33111c1d108d4d6e02a95", 28 | "url": "/frontend/build/static/media/ic_arrow-down-circle.3d1db861.svg" 29 | }, 30 | { 31 | "revision": "968dcb9920a507c65df339dca7c73edd", 32 | "url": "/frontend/build/static/media/ig_coin_card.968dcb99.png" 33 | }, 34 | { 35 | "revision": "91f341670a83880989a6a467a519f2a7", 36 | "url": "/frontend/build/static/media/ig_lose_coin.91f34167.png" 37 | }, 38 | { 39 | "revision": "44207f94d90658bedf76545bed55f30d", 40 | "url": "/frontend/build/static/media/ig_win_coin.44207f94.png" 41 | }, 42 | { 43 | "revision": "6ef855666d8f483e3f6fa8cdffb27051", 44 | "url": "/frontend/build/static/media/tt-usdt.6ef85566.png" 45 | } 46 | ]); -------------------------------------------------------------------------------- /frontend/build/service-worker.js: -------------------------------------------------------------------------------- 1 | /** 2 | * Welcome to your Workbox-powered service worker! 3 | * 4 | * You'll need to register this file in your web app and you should 5 | * disable HTTP caching for this file too. 6 | * See https://goo.gl/nhQhGp 7 | * 8 | * The rest of the code is auto-generated. Please don't update this file 9 | * directly; instead, make changes to your Workbox build configuration 10 | * and re-run your build process. 11 | * See https://goo.gl/2aRDsh 12 | */ 13 | 14 | importScripts("https://storage.googleapis.com/workbox-cdn/releases/4.3.1/workbox-sw.js"); 15 | 16 | importScripts( 17 | "/frontend/build/precache-manifest.7f0a1849234fb16afd74cd290c1ec9fc.js" 18 | ); 19 | 20 | self.addEventListener('message', (event) => { 21 | if (event.data && event.data.type === 'SKIP_WAITING') { 22 | self.skipWaiting(); 23 | } 24 | }); 25 | 26 | workbox.core.clientsClaim(); 27 | 28 | /** 29 | * The workboxSW.precacheAndRoute() method efficiently caches and responds to 30 | * requests for URLs in the manifest. 31 | * See https://goo.gl/S9QRab 32 | */ 33 | self.__precacheManifest = [].concat(self.__precacheManifest || []); 34 | workbox.precaching.precacheAndRoute(self.__precacheManifest, {}); 35 | 36 | workbox.routing.registerNavigationRoute(workbox.precaching.getCacheKeyForURL("/frontend/build/index.html"), { 37 | 38 | blacklist: [/^\/_/,/\/[^/?]+\.[^/]+$/], 39 | }); 40 | -------------------------------------------------------------------------------- /frontend/build/static/css/main.54b516df.chunk.css: -------------------------------------------------------------------------------- 1 | body{margin:0;font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Oxygen,Ubuntu,Cantarell,Fira Sans,Droid Sans,Helvetica Neue,sans-serif;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}code{font-family:source-code-pro,Menlo,Monaco,Consolas,Courier New,monospace}#root,.app,body,html{height:100%;font-family:Russo One,sans-serif}.app{background-image:url(/frontend/build/static/media/bg.03b4fd11.jpg);background-repeat:no-repeat;background-size:cover;background-position:50%;min-height:510px}.app-message{padding-top:5vh;text-align:center;color:#fff}.game-container{display:flex;max-width:420px;width:95%;align-items:center;flex-direction:column;margin:1em auto auto}.display-container{margin-bottom:1.5em;width:100%}.coin-display-container{position:relative;width:90%}.coin-display-container img{width:100%}.flip{-webkit-animation:flip .5s linear 3;animation:flip .5s linear 3}.container{text-align:center;margin-bottom:1em}.center{position:absolute;top:50%;left:50%;-webkit-transform:translate(-50%,-50%);transform:translate(-50%,-50%)}.coin{-webkit-transform-style:preserve-3d;transform-style:preserve-3d;-webkit-transform:rotateX(15deg);transform:rotateX(15deg);-webkit-transition:all .5s linear;transition:all .5s linear}.button{outline:none;border:none;height:60px;width:150px;background-color:transparent;background-size:contain;background-repeat:no-repeat;-webkit-transition:all .25s ease;transition:all .25s ease}.button[disabled]{-webkit-filter:grayscale(1);filter:grayscale(1)}.bet-button{background-image:url()}.reset-button{background-image:url()}.double-button{background-image:url()}@-webkit-keyframes flip{0%{-webkit-transform:rotateX(10deg);transform:rotateX(10deg)}to{-webkit-transform:rotateX(370deg);transform:rotateX(370deg)}}@keyframes flip{0%{-webkit-transform:rotateX(10deg);transform:rotateX(10deg)}to{-webkit-transform:rotateX(370deg);transform:rotateX(370deg)}}#debug-messages{background-image:none;background-color:#fff;font-family:source-code-pro,Menlo,Monaco,Consolas,Courier New,monospace}.coin-container{-webkit-transform-style:preserve-3d;transform-style:preserve-3d;-webkit-transition:all .5s;transition:all .5s;position:relative;width:140px;height:140px;border-radius:50%}.flipped{-webkit-transform:rotateX(180deg);transform:rotateX(180deg)}.face{-webkit-backface-visibility:hidden;backface-visibility:hidden}.top-face{-webkit-transform:translateZ(8px);transform:translateZ(8px)}.disk-body,.top-back-face{background:#444}.disk-body{width:140px;height:16px;position:absolute;-webkit-backface-visibility:hidden;backface-visibility:hidden;-webkit-transform:translate3d(0,63px,0) rotateX(90deg);transform:translate3d(0,63px,0) rotateX(90deg)}.disk-body.back{background:#938506;-webkit-transform:translate3d(0,63px,0) rotateX(90deg) rotateY(180deg);transform:translate3d(0,63px,0) rotateX(90deg) rotateY(180deg)}.disk{overflow:hidden;border-radius:50%;position:absolute;height:100%;width:100%;background:#444}.bottom-front-face{background-size:contain;background-image:url(/frontend/build/static/media/ig_lose_coin.91f34167.png);-webkit-transform:translateZ(-8px) rotateX(180deg);transform:translateZ(-8px) rotateX(180deg);box-shadow:0 0 8px rgba(0,0,0,.45)}.bottom-back-face{-webkit-transform:translateZ(-8px);transform:translateZ(-8px);background:#444}.half-face-lower{top:50%;background:#938506}.half-face-upper{background:#444}.half-face{position:absolute;height:50%;width:100%}.top-face-front{background-size:contain;background-image:url(/frontend/build/static/media/ig_win_coin.44207f94.png)}.display{flex-wrap:wrap;justify-content:space-between;width:100%}.display,.title-container{display:flex;flex-direction:row}.title-container{font-size:1.25em;color:#fff;margin:8px}.value{margin-left:6px}@media (max-width:400px){.title-container{font-size:1em}}.coin-picker-container{display:flex;width:100%}.coin-button-container{flex:1 1;padding:6px 12px}.coin-picker-button-icon{width:20px;height:20px;margin-right:6px}.coin-picker-button{width:100%;height:100%;outline:none;-webkit-filter:grayscale(100%);filter:grayscale(100%);display:flex;align-items:center;justify-content:center;min-height:42px;font-family:inherit;cursor:pointer;border-radius:8px;color:#fff;font-size:16px;font-weight:700;border:3px solid #c5c5c5;background-color:#dedede;-webkit-transition:all .25s ease;transition:all .25s ease}.coin-picker-button.active{cursor:default;-webkit-filter:grayscale(0);filter:grayscale(0);border-color:#fe9f25;background-color:#ffc529}.header{z-index:2000;padding:1em 1.5em;background:#48197c;color:#fff;justify-content:space-between}.header,.header-reward{display:flex;align-items:center}.header-reward{justify-content:center}.reward-container{border-radius:5px;margin:0 8px;background:#2a0e4a;padding:2px 8px}.arrow-circle{-webkit-transition:all .25s;transition:all .25s}.arrow-circle.open{-webkit-transform:rotateX(180deg);transform:rotateX(180deg)}.ReferralRule-container{top:55px;bottom:0;width:100%;overflow:auto;position:absolute;z-index:200;background:rgba(29,6,34,.9)}.inner-container{max-width:420px;margin:auto;color:#fff;padding:24px}.share-container{line-height:1.5;font-size:1.15em;text-align:center;border-radius:5px;background:rgba(72,26,86,.75);padding:24px;margin-bottom:1.5em}.referral-title{text-align:center;color:#faf528;margin-bottom:1em}.referral-link-container{display:flex;align-items:stretch;justify-content:center;margin-bottom:1.5em}.referral-link{font-size:.85em;font-family:sans-serif;word-break:break-all;padding:12px;background:rgba(56,34,63,.75);border-radius:5px;flex:1 1}.copy-button{cursor:pointer;margin-bottom:6px;border:none;outline:none;font-size:1.5em;font-family:inherit;padding:20px;margin-left:8px;border-radius:5px;color:#000;background:#ffc529;box-shadow:0 6px 2px #ba751a;-webkit-transition:all .25s;transition:all .25s}.copy-button.clicked{margin-top:6px;margin-bottom:0;box-shadow:0 0 0 #ba751a}.referral-stats-container{margin-bottom:2em}.referral-stat{display:flex;justify-content:space-between;margin-bottom:.75em;font-size:.85em}.referral-rule-block,.table{margin-bottom:1em}.table{display:flex;border-radius:5px;overflow:hidden}.table-column{flex:1 1}.table-column.dark{background-color:#461351}.table-column.light{background-color:#863499}.table-column.med{background-color:#5d226b}.cell{display:flex;padding:6px;align-items:center;justify-content:center;height:50px;text-align:center;border:1px solid hsla(0,0%,100%,.05)} 2 | /*# sourceMappingURL=main.54b516df.chunk.css.map */ -------------------------------------------------------------------------------- /frontend/build/static/css/main.54b516df.chunk.css.map: -------------------------------------------------------------------------------- 1 | {"version":3,"sources":["index.css","App.css","Game.css","Coin.css","Display.css","CoinPicker.css","Header.css","ReferralRules.css"],"names":[],"mappings":"AAAA,KACE,QAAS,CACT,mIAEY,CACZ,kCAAmC,CACnC,iCACF,CAEA,KACE,uEAEF,CCZA,qBACE,WAAY,CACZ,gCACF,CAEA,KACE,kEAA4C,CAC5C,2BAA4B,CAC5B,qBAAsB,CACtB,uBAA2B,CAC3B,gBACF,CAEA,aACE,eAAgB,CAChB,iBAAkB,CAClB,UACF,CCjBA,gBACE,YAAa,CACb,eAAgB,CAChB,SAAU,CAEV,kBAAmB,CACnB,qBAAsB,CACtB,oBACF,CAEA,mBACE,mBAAoB,CACpB,UACF,CAEA,wBACE,iBAAkB,CAClB,SACF,CAEA,4BACE,UACF,CAEA,MACE,mCAA6B,CAA7B,2BACF,CAEA,WACE,iBAAkB,CAClB,iBACF,CAEA,QACE,iBAAkB,CAClB,OAAQ,CACR,QAAS,CACT,sCAA8B,CAA9B,8BACF,CAEA,MACE,mCAA4B,CAA5B,2BAA4B,CAC5B,gCAAyB,CAAzB,wBAAyB,CACzB,iCAA2B,CAA3B,yBACF,CAEA,QACE,YAAa,CACb,WAAY,CACZ,WAAY,CACZ,WAAY,CACZ,4BAA6B,CAC7B,uBAAwB,CACxB,2BAA4B,CAC5B,gCAA0B,CAA1B,wBACF,CAEA,kBACE,2BAAoB,CAApB,mBACF,CAEA,YACE,4iKACF,CAGA,cACE,4wJACF,CAEA,eACE,wrJACF,CAGA,wBACE,GACE,gCAAyB,CAAzB,wBACF,CACA,GACE,iCAA0B,CAA1B,yBACF,CACF,CAPA,gBACE,GACE,gCAAyB,CAAzB,wBACF,CACA,GACE,iCAA0B,CAA1B,yBACF,CACF,CAEA,gBACE,qBAAsB,CACtB,qBAAuB,CACvB,uEACF,CCxFA,gBACE,mCAA4B,CAA5B,2BAA4B,CAC5B,0BAAoB,CAApB,kBAAoB,CACpB,iBAAkB,CAClB,WAAY,CACZ,YAAa,CACb,iBACF,CAEA,SACE,iCAA0B,CAA1B,yBACF,CAEA,MACE,kCAA2B,CAA3B,0BACF,CAEA,UACE,iCAA0B,CAA1B,yBACF,CAMA,0BAHE,eAUF,CAPA,WACE,WAAY,CACZ,WAAY,CACZ,iBAAkB,CAElB,kCAA2B,CAA3B,0BAA2B,CAC3B,sDAAgD,CAAhD,8CACF,CAEA,gBACE,kBAAmB,CACnB,sEAAgE,CAAhE,8DACF,CAEA,MACE,eAAgB,CAChB,iBAAkB,CAClB,iBAAkB,CAClB,WAAY,CACZ,UAAW,CACX,eACF,CAEA,mBACE,uBAAwB,CACxB,4EAAsD,CACtD,kDAA2C,CAA3C,0CAA2C,CAC3C,kCACF,CAEA,kBACE,kCAA2B,CAA3B,0BAA2B,CAC3B,eACF,CAEA,iBACE,OAAQ,CACR,kBACF,CAEA,iBACE,eACF,CAEA,WACE,iBAAkB,CAClB,UAAW,CACX,UACF,CAEA,gBACE,uBAAwB,CACxB,2EACF,CC9EA,SAGE,cAAe,CACf,6BAA8B,CAC9B,UACF,CAEA,0BAPE,YAAa,CACb,kBAYF,CANA,iBACE,gBAAiB,CACjB,UAAc,CACd,UAGF,CAEA,OACE,eACF,CAEA,yBACE,iBACE,aACF,CACF,CCxBA,uBACE,YAAa,CACb,UACF,CAEA,uBACE,QAAO,CACP,gBACF,CAEA,yBACE,UAAW,CACX,WAAY,CACZ,gBACF,CAEA,oBACE,UAAW,CACX,WAAY,CACZ,YAAa,CACb,8BAAuB,CAAvB,sBAAuB,CACvB,YAAa,CACb,kBAAmB,CACnB,sBAAuB,CACvB,eAAgB,CAChB,mBAAoB,CACpB,cAAe,CACf,iBAAkB,CAClB,UAAY,CACZ,cAAe,CACf,eAAiB,CACjB,wBAAkC,CAClC,wBAAkC,CAClC,gCAAyB,CAAzB,wBACF,CAEA,2BACE,cAAe,CACf,2BAAqB,CAArB,mBAAqB,CACrB,oBAA6B,CAC7B,wBACF,CCzCA,QACE,YAAa,CACb,iBAAkB,CAClB,kBAA0B,CAC1B,UAAY,CAEZ,6BAEF,CAEA,uBALE,YAAa,CAEb,kBAOF,CAJA,eAEE,sBAEF,CAEA,kBACE,iBAAkB,CAClB,YAAa,CACb,kBAAyB,CACzB,eACF,CAEA,cACE,2BAAqB,CAArB,mBACF,CAEA,mBACE,iCAA0B,CAA1B,yBACF,CC7BA,wBACE,QAAQ,CACR,QAAQ,CACR,UAAW,CACX,aAAc,CACd,iBAAkB,CAClB,WAAY,CACZ,2BACF,CAEA,iBACE,eAAgB,CAChB,WAAY,CACZ,UAAY,CACZ,YACF,CAEA,iBACE,eAAgB,CAChB,gBAAiB,CACjB,iBAAkB,CAClB,iBAAkB,CAClB,6BAA+B,CAC/B,YAAa,CACb,mBACF,CAEA,gBACE,iBAAkB,CAClB,aAAsB,CACtB,iBACF,CAEA,yBACE,YAAa,CACb,mBAAoB,CACpB,sBAAuB,CACvB,mBACF,CAEA,eACE,eAAgB,CAChB,sBAAuB,CACvB,oBAAqB,CACrB,YAAa,CACb,6BAAgC,CAChC,iBAAkB,CAClB,QACF,CAEA,aACE,cAAe,CACf,iBAAkB,CAClB,WAAY,CACZ,YAAa,CACb,eAAgB,CAChB,mBAAoB,CACpB,YAAa,CACb,eAAgB,CAChB,iBAAkB,CAClB,UAAY,CACZ,kBAA2B,CAC3B,4BAAqC,CACrC,2BAAqB,CAArB,mBACF,CAEA,qBACE,cAAe,CACf,eAAgB,CAChB,wBACF,CAEA,0BACE,iBACF,CAEA,eACE,YAAa,CACb,6BAA8B,CAC9B,mBAAqB,CACrB,eACF,CAMA,4BAHE,iBAQF,CALA,OACE,YAAa,CACb,iBAAkB,CAClB,eAEF,CAEA,cACE,QACF,CAEA,mBACE,wBACF,CAEA,oBACE,wBACF,CAEA,kBACE,wBACF,CAEA,MACE,YAAa,CACb,WAAY,CACZ,kBAAmB,CACnB,sBAAuB,CACvB,WAAY,CACZ,iBAAkB,CAClB,oCACF","file":"main.54b516df.chunk.css","sourcesContent":["body {\n margin: 0;\n font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', 'Oxygen',\n 'Ubuntu', 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue',\n sans-serif;\n -webkit-font-smoothing: antialiased;\n -moz-osx-font-smoothing: grayscale;\n}\n\ncode {\n font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New',\n monospace;\n}\n","html, body, .app, #root {\n height: 100%;\n font-family: 'Russo One', sans-serif;\n}\n\n.app {\n background-image: url(\"../../assets/bg.jpg\");\n background-repeat: no-repeat;\n background-size: cover;\n background-position: center;\n min-height: 510px;\n}\n\n.app-message {\n padding-top: 5vh;\n text-align: center;\n color: white;\n}\n",".game-container {\n display: flex;\n max-width: 420px;\n width: 95%;\n margin: auto;\n align-items: center;\n flex-direction: column;\n margin-top: 1em;\n}\n\n.display-container {\n margin-bottom: 1.5em;\n width: 100%;\n}\n\n.coin-display-container {\n position: relative;\n width: 90%;\n}\n\n.coin-display-container img {\n width: 100%;\n}\n\n.flip {\n animation: flip 0.5s linear 3;\n}\n\n.container {\n text-align: center;\n margin-bottom: 1em;\n}\n\n.center {\n position: absolute;\n top: 50%;\n left: 50%;\n transform: translate(-50%,-50%)\n}\n\n.coin {\n transform-style: preserve-3d;\n transform: rotateX(15deg);\n transition: all 0.5s linear;\n}\n\n.button {\n outline: none;\n border: none;\n height: 60px;\n width: 150px;\n background-color: transparent;\n background-size: contain;\n background-repeat: no-repeat;\n transition: all 0.25s ease;\n}\n\n.button[disabled] {\n filter: grayscale(1);\n}\n\n.bet-button {\n background-image: url(\"../../assets/ig_bet_btn.png\");\n}\n\n\n.reset-button {\n background-image: url(\"../../assets/ig_reset_btn.png\");\n}\n\n.double-button {\n background-image: url(\"../../assets/ig_double_btn.png\");\n}\n\n\n@keyframes flip {\n 0% {\n transform: rotateX(10deg);\n }\n 100% {\n transform: rotateX(370deg);\n }\n}\n\n#debug-messages {\n background-image: none;\n background-color: white;\n font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New', monospace;\n}\n",".coin-container {\n transform-style: preserve-3d;\n transition: 0.5s all;\n position: relative;\n width: 140px;\n height: 140px;\n border-radius: 50%;\n}\n\n.flipped {\n transform: rotateX(180deg);\n}\n\n.face {\n backface-visibility: hidden;\n}\n\n.top-face {\n transform: translateZ(8px);\n}\n\n.top-back-face {\n background: #444;\n}\n\n.disk-body {\n width: 140px;\n height: 16px;\n position: absolute;\n background: #444;\n backface-visibility: hidden;\n transform: translate3d(0,63px,0) rotateX(90deg);\n}\n\n.disk-body.back {\n background: #938506;\n transform: translate3d(0,63px,0) rotateX(90deg) rotateY(180deg);\n}\n\n.disk {\n overflow: hidden;\n border-radius: 50%;\n position: absolute;\n height: 100%;\n width: 100%;\n background: #444;\n}\n\n.bottom-front-face {\n background-size: contain;\n background-image: url(\"../../assets/ig_lose_coin.png\");\n transform: translateZ(-8px) rotateX(180deg);\n box-shadow: 0 0 8px rgba(0, 0, 0, 0.45);\n}\n\n.bottom-back-face {\n transform: translateZ(-8px);\n background: #444;\n}\n\n.half-face-lower {\n top: 50%;\n background: #938506;\n}\n\n.half-face-upper {\n background: #444;\n}\n\n.half-face {\n position: absolute;\n height: 50%;\n width: 100%;\n}\n\n.top-face-front {\n background-size: contain;\n background-image: url(\"../../assets/ig_win_coin.png\");\n}\n",".display {\n display: flex;\n flex-direction: row;\n flex-wrap: wrap;\n justify-content: space-between;\n width: 100%;\n}\n\n.title-container {\n font-size: 1.25em;\n color: #ffffff;\n margin: 8px;\n display: flex;\n flex-direction: row;\n}\n\n.value {\n margin-left: 6px;\n}\n\n@media (max-width: 400px) {\n .title-container {\n font-size: 1em;\n }\n}\n",".coin-picker-container {\n display: flex;\n width: 100%;\n}\n\n.coin-button-container {\n flex: 1;\n padding: 6px 12px;\n}\n\n.coin-picker-button-icon {\n width: 20px;\n height: 20px;\n margin-right: 6px;\n}\n\n.coin-picker-button {\n width: 100%;\n height: 100%;\n outline: none;\n filter: grayscale(100%);\n display: flex;\n align-items: center;\n justify-content: center;\n min-height: 42px;\n font-family: inherit;\n cursor: pointer;\n border-radius: 8px;\n color: white;\n font-size: 16px;\n font-weight: bold;\n border: 3px solid rgb(197,197,197);\n background-color: rgb(222,222,222);\n transition: 0.25s ease all\n}\n\n.coin-picker-button.active {\n cursor: default;\n filter: grayscale(0%);\n border-color: rgb(254,159,37);\n background-color: rgb(255,197,41);\n}\n",".header {\n z-index: 2000;\n padding: 1em 1.5em;\n background: rgb(72,25,124);\n color: white;\n display: flex;\n justify-content: space-between;\n align-items: center;\n}\n\n.header-reward {\n display: flex;\n justify-content: center;\n align-items: center;\n}\n\n.reward-container {\n border-radius: 5px;\n margin: 0 8px;\n background: rgb(42,14,74);\n padding: 2px 8px;\n}\n\n.arrow-circle {\n transition: 0.25s all;\n}\n\n.arrow-circle.open {\n transform: rotateX(180deg);\n}\n",".ReferralRule-container {\n top:55px;\n bottom:0;\n width: 100%;\n overflow: auto;\n position: absolute;\n z-index: 200;\n background: rgba(29,6,34,0.9);\n}\n\n.inner-container {\n max-width: 420px;\n margin: auto;\n color: white;\n padding: 24px;\n}\n\n.share-container {\n line-height: 1.5;\n font-size: 1.15em;\n text-align: center;\n border-radius: 5px;\n background: rgba(72,26,86,0.75);\n padding: 24px;\n margin-bottom: 1.5em;\n}\n\n.referral-title {\n text-align: center;\n color: rgb(250,245,40);\n margin-bottom: 1em;\n}\n\n.referral-link-container {\n display: flex;\n align-items: stretch;\n justify-content: center;\n margin-bottom: 1.5em;\n}\n\n.referral-link {\n font-size: .85em;\n font-family: sans-serif;\n word-break: break-all;\n padding: 12px;\n background: rgba(56,34,63, 0.75);\n border-radius: 5px;\n flex: 1;\n}\n\n.copy-button {\n cursor: pointer;\n margin-bottom: 6px;\n border: none;\n outline: none;\n font-size: 1.5em;\n font-family: inherit;\n padding: 20px;\n margin-left: 8px;\n border-radius: 5px;\n color: black;\n background: rgb(255,197,41);\n box-shadow: 0 6px 2px rgb(186,117,26);\n transition: 0.25s all;\n}\n\n.copy-button.clicked {\n margin-top: 6px;\n margin-bottom: 0;\n box-shadow: 0 0 0 rgb(186,117,26);\n}\n\n.referral-stats-container {\n margin-bottom: 2em;\n}\n\n.referral-stat {\n display: flex;\n justify-content: space-between;\n margin-bottom: 0.75em;\n font-size: 0.85em;\n}\n\n.referral-rule-block {\n margin-bottom: 1em;\n}\n\n.table {\n display: flex;\n border-radius: 5px;\n overflow: hidden;\n margin-bottom: 1em;\n}\n\n.table-column {\n flex: 1;\n}\n\n.table-column.dark {\n background-color: rgb(70,19,81);\n}\n\n.table-column.light {\n background-color: rgb(134,52,153);\n}\n\n.table-column.med {\n background-color: rgb(93,34,107);\n}\n\n.cell {\n display: flex;\n padding: 6px;\n align-items: center;\n justify-content: center;\n height: 50px;\n text-align: center;\n border: 1px solid rgba(255,255,255,0.05);\n}\n"]} -------------------------------------------------------------------------------- /frontend/build/static/js/main.22e40df3.chunk.js: -------------------------------------------------------------------------------- 1 | (window.webpackJsonpfrontend=window.webpackJsonpfrontend||[]).push([[0],{126:function(e){e.exports=JSON.parse('{"a":[{"constant":true,"inputs":[{"name":"data","type":"uint256[]"}],"name":"sum","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"pure","type":"function"},{"constant":true,"inputs":[{"name":"addr","type":"address"}],"name":"hasReferrer","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"_secondsUntilInactive","type":"uint256"}],"name":"setSecondsUntilInactive","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"getTime","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"name":"","type":"address"}],"name":"accounts","outputs":[{"name":"referrer","type":"address"},{"name":"reward","type":"uint256"},{"name":"referredCount","type":"uint256"},{"name":"lastActiveTimestamp","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[],"name":"renounceOwnership","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[{"name":"amount","type":"uint256"}],"name":"getRefereeBonusRate","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"owner","outputs":[{"name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"isOwner","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"_onlyRewardActiveReferrers","type":"bool"}],"name":"setOnlyRewardAActiveReferrers","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"inputs":[{"name":"_decimals","type":"uint256"},{"name":"_referralBonus","type":"uint256"},{"name":"_secondsUntilInactive","type":"uint256"},{"name":"_onlyRewardActiveReferrers","type":"bool"},{"name":"_levelRate","type":"uint256[]"},{"name":"_refereeBonusRateMap","type":"uint256[]"},{"name":"_tokenAddresses","type":"address[]"},{"name":"_randomNumberContractAddress","type":"address"}],"payable":false,"stateMutability":"nonpayable","type":"constructor"},{"payable":true,"stateMutability":"payable","type":"fallback"},{"anonymous":false,"inputs":[{"indexed":false,"name":"player","type":"address"},{"indexed":false,"name":"winnings","type":"uint256"}],"name":"BetSettled","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"name":"referee","type":"address"},{"indexed":false,"name":"referrer","type":"address"}],"name":"RegisteredReferer","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"name":"referee","type":"address"},{"indexed":false,"name":"referrer","type":"address"},{"indexed":false,"name":"reason","type":"string"}],"name":"RegisteredRefererFailed","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"name":"from","type":"address"},{"indexed":false,"name":"to","type":"address"},{"indexed":false,"name":"amount","type":"uint256"},{"indexed":false,"name":"level","type":"uint256"}],"name":"PaidReferral","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"name":"user","type":"address"},{"indexed":false,"name":"timestamp","type":"uint256"}],"name":"UpdatedUserLastActiveTime","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"previousOwner","type":"address"},{"indexed":true,"name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"constant":false,"inputs":[{"name":"_tokenAddress","type":"address"}],"name":"addToken","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_tokenAddress","type":"address"}],"name":"deleteToken","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_referrer","type":"address"}],"name":"bet","outputs":[],"payable":true,"stateMutability":"payable","type":"function"},{"constant":false,"inputs":[],"name":"bet","outputs":[],"payable":true,"stateMutability":"payable","type":"function"},{"constant":false,"inputs":[{"name":"","type":"address"},{"name":"_value","type":"uint256"},{"name":"_data","type":"bytes"}],"name":"onTokenTransfer","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_amount","type":"uint256"}],"name":"withdraw","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_tokenAddress","type":"address"},{"name":"_amount","type":"uint256"}],"name":"withdrawTokens","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"}]}')},129:function(e,t,a){e.exports=a.p+"static/media/ig_coin_card.968dcb99.png"},130:function(e,t,a){e.exports=a.p+"static/media/tt-usdt.6ef85566.png"},131:function(e,t){e.exports=""},133:function(e){e.exports=JSON.parse('{"a":[{"constant":false,"inputs":[{"name":"spender","type":"address"},{"name":"value","type":"uint256"}],"name":"approve","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"totalSupply","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"sender","type":"address"},{"name":"recipient","type":"address"},{"name":"amount","type":"uint256"}],"name":"transferFrom","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"spender","type":"address"},{"name":"addedValue","type":"uint256"}],"name":"increaseAllowance","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[{"name":"account","type":"address"}],"name":"balanceOf","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"spender","type":"address"},{"name":"subtractedValue","type":"uint256"}],"name":"decreaseAllowance","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"decimals","outputs":[{"name":"","type":"uint8"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"recipient","type":"address"},{"name":"amount","type":"uint256"}],"name":"transfer","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[{"name":"owner","type":"address"},{"name":"spender","type":"address"}],"name":"allowance","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"anonymous":false,"inputs":[{"indexed":true,"name":"from","type":"address"},{"indexed":true,"name":"to","type":"address"},{"indexed":false,"name":"value","type":"uint256"},{"indexed":false,"name":"data","type":"bytes"}],"name":"Transfer","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"from","type":"address"},{"indexed":true,"name":"to","type":"address"},{"indexed":false,"name":"value","type":"uint256"}],"name":"Transfer","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"owner","type":"address"},{"indexed":true,"name":"spender","type":"address"},{"indexed":false,"name":"value","type":"uint256"}],"name":"Approval","type":"event"},{"constant":false,"inputs":[{"name":"","type":"address"},{"name":"","type":"uint256"},{"name":"","type":"bytes"}],"name":"transferAndCall","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"nonpayable","type":"function"}]}')},135:function(e,t,a){e.exports=a(291)},140:function(e,t,a){},141:function(e,t,a){},143:function(e,t){},147:function(e,t){},176:function(e,t){},178:function(e,t){},282:function(e,t,a){},283:function(e,t,a){},284:function(e,t,a){},285:function(e,t,a){},286:function(e,t,a){},290:function(e,t,a){},291:function(e,t,a){"use strict";a.r(t);var n=a(0),r=a.n(n),s=a(125),i=a.n(s),o=(a(140),a(19)),c=a(4),l=a(5),u=a(7),d=a(6),p=(a(141),a(126)),m=a(17),f=a(134),y=a(127),b=r.a.createContext({enable:function(){},address:"",signer:void 0,enabling:!1,enabled:!1}),v=b.Consumer,g=b.Provider,h=function(e){Object(u.a)(a,e);var t=Object(d.a)(a);function a(e){var n;Object(c.a)(this,a),(n=t.call(this,e)).enable=function(){n.state.signer&&(n.setState({enabling:!0}),n.state.signer.provider._web3Provider.enable().then(function(e){var t=Object(f.a)(e,1)[0];n.setState({enabling:!1,enabled:!0,address:t})}).catch(function(){n.setState({enabling:!1,enabled:!1})}))};var r=!!window.ethereum;return n.state={address:"",signer:r&&new y.Web3Provider(window.ethereum).getSigner(),enabled:!1,enabling:r&&!!e.enableOnLoad},n}return Object(l.a)(a,[{key:"componentDidMount",value:function(){this.props.enableOnLoad&&this.enable()}},{key:"render",value:function(){return r.a.createElement(g,{value:Object(m.a)(Object(m.a)({},this.state),{},{enable:this.enable})},this.props.children)}}]),a}(r.a.PureComponent),E=v,w=function(e){Object(u.a)(a,e);var t=Object(d.a)(a);function a(){var e;Object(c.a)(this,a);for(var n=arguments.length,r=new Array(n),s=0;s=0||(r[a]=e[a]);return r}(e,t);if(Object.getOwnPropertySymbols){var s=Object.getOwnPropertySymbols(e);for(n=0;n=0||Object.prototype.propertyIsEnumerable.call(e,a)&&(r[a]=e[a])}return r}var L,Q=r.a.createElement("g",{id:"ic_arrow-down-circle",transform:"translate(1,1)",fill:"none",stroke:"#FFFFFF",strokeWidth:2},r.a.createElement("circle",{id:"Oval",cx:10,cy:10,r:10}),r.a.createElement("polyline",{id:"Path",points:"6 10 10 14 14 10"}),r.a.createElement("path",{d:"M10,6 L10,14",id:"Path"})),q=function(e){var t=e.svgRef,a=e.title,n=V(e,["svgRef","title"]);return r.a.createElement("svg",Z({width:"22px",height:"22px",viewBox:"0 0 22 22",ref:t},n),r.a.createElement("title",null,a),Q)},W=r.a.forwardRef(function(e,t){return r.a.createElement(q,Z({svgRef:t},e))}),Y=(a.p,a(132)),K=(a(290),function(e){Object(u.a)(a,e);var t=Object(d.a)(a);function a(){var e;Object(c.a)(this,a);for(var n=arguments.length,r=new Array(n),s=0;s="5"?t.slice(a,n-1)+(parseInt(t.slice(n-1,n))+1).toString():t.slice(a,n);return r=r.replace(/0+$/,""),t.slice(0,a)+(r?"."+r:"")}},{key:"render",value:function(){var e=this.state.open;return r.a.createElement(r.a.Fragment,null,r.a.createElement("div",{className:"header"},r.a.createElement("div",null,"Double or Nothing"),r.a.createElement("div",{className:"header-reward"},r.a.createElement("div",null,"Reward"),r.a.createElement("div",{className:"reward-container"},this.calcualteEther(this.props.rewards)),r.a.createElement(W,{onClick:this.toggle,className:"arrow-circle ".concat(e?"open":"")}))),e&&r.a.createElement(K,{address:this.props.address,reward:this.calcualteEther(this.props.rewards),refereeCount:this.props.refereeCount,activeTimestamp:this.props.activeTimestamp}))}}]),a}(r.a.PureComponent),z=.5,_=function(e){Object(u.a)(a,e);var t=Object(d.a)(a);function a(e){var n;Object(c.a)(this,a),(n=t.call(this,e)).intervals=[],n.watchBalance=function(){n.intervals.push(setInterval(function(){n.props.contract.provider.getBalance(n.props.address).then(function(e){n.setState({balance:e.toString()})})},500))},n.watchReferrals=function(){n.intervals.push(setInterval(function(){n.props.contract.accounts(n.props.address).then(function(e){n.setState({accountInfo:{reward:e.reward.toString(),referredCount:e.referredCount.toString(),activeTimestamp:1e3*e.lastActiveTimestamp.toString()+864e5}})})},500))},n.log=function(e){n.setState(function(t){return{debugMessages:["".concat(e)].concat(t.debugMessages)}})},n.play=function(e){var t;n.setState({disabled:!0});var a=[{value:Object(T.parseEther)(e.toString()).toHexString(),gasPrice:Object(T.bigNumberify)("1000000000").toHexString(),gasLimit:Object(T.bigNumberify)("2000000").toHexString()}],r=!!n.state.referrer;r&&a.unshift(n.state.referrer),(t=n.props.contract)[r?"bet(address)":"bet()"].apply(t,a).then(function(e){return n.setState({flipping:!0}),e.wait()}).then(function(e){var t=e.events.find(function(e){return"BetSettled"===e.event});n.setState({flipping:!1,win:"0"!==t.args.winnings.toString()})}).catch(function(e){console.log('setState("logs", '.concat(e,")")),n.log(JSON.stringify(e))}).finally(function(){n.setState({disabled:!1,flipping:!1})})},n.playGame=function(){n.play(n.state.betAmount)},n.double=function(){var e=2*n.state.betAmount;n.setState({betAmount:e}),n.play(e)},n.reset=function(){n.setState({betAmount:z})},n.watchBalance(),n.watchReferrals();var r=window.location.search.match(/referrer=(0x[a-fA-F0-9]{40})/),s=r?r[1]:"";return n.state={referrer:s.toLowerCase(),betAmount:z,balance:"",win:!0,disabled:!1,flipping:!1,accountInfo:{reward:"0",referredCount:"0",activeTimestamp:Date.now()},debugMessages:[]},n}return Object(l.a)(a,[{key:"componentWillUnmount",value:function(){this.intervals.forEach(function(e){return clearInterval(e)})}},{key:"render",value:function(){var e=this.state,t=e.disabled,a=e.flipping,n=e.win,s=e.balance,i=e.betAmount,o=e.accountInfo;return r.a.createElement(r.a.Fragment,null,r.a.createElement(X,{address:this.props.address,rewards:o.reward,refereeCount:o.referredCount,activeTimestamp:o.activeTimestamp}),r.a.createElement(G,{changeContract:this.props.changeContract,active:S.DoubleOrNothing,double:this.double,playGame:this.playGame,reset:this.reset,betAmount:i,balance:s,win:n,disabled:t,flipping:a,decimals:18,symbol:"TT"}),r.a.createElement("div",{id:"debug-messages"},this.state.debugMessages.map(function(e,t){return r.a.createElement("div",{key:t},e)})))}}]),a}(r.a.PureComponent),$=.5,ee=function(e){Object(u.a)(a,e);var t=Object(d.a)(a);function a(e){var n;return Object(c.a)(this,a),(n=t.call(this,e)).interval=void 0,n.watchBalance=function(){n.interval=setInterval(function(){n.props.contract.balanceOf(n.props.address).then(function(e){return n.props.contract.decimals().then(function(t){n.setState({balance:e.toString(),decimals:t})})})},500)},n.play=function(e){n.setState({disabled:!0}),n.props.contract.decimals().then(function(t){console.log(t),n.props.contract.transferAndCall(n.props.gameAddress,Object(T.parseUnits)(e.toString(),t),"0xfae",{gasPrice:Object(T.bigNumberify)("1000000000").toHexString(),gasLimit:Object(T.bigNumberify)("100000").toHexString()}).then(function(e){return n.setState({flipping:!0}),e.wait()}).then(function(e){var t=!!e.events.find(function(e){return"Transfer"===e.event&&e.args.to.toLowerCase()===n.props.address.toLowerCase()});n.setState({flipping:!1,win:t})}).finally(function(){n.setState({disabled:!1,flipping:!1})})})},n.playGame=function(){n.play(n.state.betAmount)},n.double=function(){var e=2*n.state.betAmount;n.setState({betAmount:e}),n.play(e)},n.reset=function(){n.setState({betAmount:$})},n.state={betAmount:$,balance:"",win:!0,disabled:!1,flipping:!1,accountInfo:{reward:"0",referredCount:"0",activeTimestamp:Date.now()},decimals:6},n.watchBalance(),n}return Object(l.a)(a,[{key:"componentWillUnmount",value:function(){this.interval&&clearInterval(this.interval)}},{key:"render",value:function(){var e=this.state,t=e.disabled,a=e.flipping,n=e.win,s=e.balance,i=e.betAmount,o=e.accountInfo,c=e.decimals;return r.a.createElement(r.a.Fragment,null,r.a.createElement(X,{address:this.props.address,rewards:o.reward,refereeCount:o.referredCount,activeTimestamp:o.activeTimestamp}),r.a.createElement(G,{changeContract:this.props.changeContract,active:this.props.contractName,double:this.double,playGame:this.playGame,reset:this.reset,betAmount:i,balance:s,win:n,disabled:t,flipping:a,decimals:c,symbol:"USDT"}))}}]),a}(r.a.PureComponent),te=function(e){Object(u.a)(a,e);var t=Object(d.a)(a);function a(){var e;Object(c.a)(this,a);for(var n=arguments.length,r=new Array(n),s=0;s 2 | 3 | 4 | 5 | 6 | 7 | 8 | -------------------------------------------------------------------------------- /frontend/build/static/media/ig_coin_card.968dcb99.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thundercore/DoubleOrNothing/dd0501343022f2c4f5217c42a69fad2280306e84/frontend/build/static/media/ig_coin_card.968dcb99.png -------------------------------------------------------------------------------- /frontend/build/static/media/ig_lose_coin.91f34167.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thundercore/DoubleOrNothing/dd0501343022f2c4f5217c42a69fad2280306e84/frontend/build/static/media/ig_lose_coin.91f34167.png -------------------------------------------------------------------------------- /frontend/build/static/media/ig_win_coin.44207f94.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thundercore/DoubleOrNothing/dd0501343022f2c4f5217c42a69fad2280306e84/frontend/build/static/media/ig_win_coin.44207f94.png -------------------------------------------------------------------------------- /frontend/build/static/media/tt-usdt.6ef85566.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thundercore/DoubleOrNothing/dd0501343022f2c4f5217c42a69fad2280306e84/frontend/build/static/media/tt-usdt.6ef85566.png -------------------------------------------------------------------------------- /frontend/environment.yml: -------------------------------------------------------------------------------- 1 | name: double-or-nothing 2 | channels: 3 | - conda-forge 4 | dependencies: 5 | - nodejs=10.16.3 # hub-analytics-frontend 6 | - yarn=1.22.0 7 | -------------------------------------------------------------------------------- /frontend/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "frontend", 3 | "version": "0.1.0", 4 | "private": true, 5 | "dependencies": { 6 | "ethers": "^4.0.29", 7 | "react": "^16.8.6", 8 | "react-countdown-now": "^2.1.1", 9 | "react-dom": "^16.8.6" 10 | }, 11 | "devDependencies": { 12 | "@types/jest": "24.0.13", 13 | "@types/node": "12.0.7", 14 | "@types/react": "16.8.19", 15 | "@types/react-dom": "16.8.4", 16 | "react-scripts": "3.1.2", 17 | "typescript": "^3.8.3" 18 | }, 19 | "scripts": { 20 | "start": "react-scripts start", 21 | "build": "PUBLIC_URL=/frontend/build react-scripts build", 22 | "test": "react-scripts test", 23 | "eject": "react-scripts eject" 24 | }, 25 | "eslintConfig": { 26 | "extends": "react-app" 27 | }, 28 | "browserslist": { 29 | "production": [ 30 | ">0.2%", 31 | "not dead", 32 | "not op_mini all" 33 | ], 34 | "development": [ 35 | "last 1 chrome version", 36 | "last 1 firefox version", 37 | "last 1 safari version" 38 | ] 39 | } 40 | } 41 | -------------------------------------------------------------------------------- /frontend/public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thundercore/DoubleOrNothing/dd0501343022f2c4f5217c42a69fad2280306e84/frontend/public/favicon.ico -------------------------------------------------------------------------------- /frontend/public/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 13 | 14 | 23 | DoubleOrNothing 24 | 25 | 26 | 27 |
28 | 38 | 39 | 40 | -------------------------------------------------------------------------------- /frontend/public/manifest.json: -------------------------------------------------------------------------------- 1 | { 2 | "short_name": "DON", 3 | "name": "Double Or Nothing", 4 | "icons": [ 5 | { 6 | "src": "favicon.ico", 7 | "sizes": "64x64 32x32 24x24 16x16", 8 | "type": "image/x-icon" 9 | } 10 | ], 11 | "start_url": ".", 12 | "display": "standalone", 13 | "theme_color": "#000000", 14 | "background_color": "#ffffff" 15 | } 16 | -------------------------------------------------------------------------------- /frontend/src/ContractEnum.ts: -------------------------------------------------------------------------------- 1 | export enum ContractEnum { 2 | TTUsdt = 'tt-usdt', 3 | DoubleOrNothing = 'doubleOrNothing' 4 | } 5 | -------------------------------------------------------------------------------- /frontend/src/assets/bg.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thundercore/DoubleOrNothing/dd0501343022f2c4f5217c42a69fad2280306e84/frontend/src/assets/bg.jpg -------------------------------------------------------------------------------- /frontend/src/assets/ic_arrow-down-circle.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | -------------------------------------------------------------------------------- /frontend/src/assets/icon_TT.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thundercore/DoubleOrNothing/dd0501343022f2c4f5217c42a69fad2280306e84/frontend/src/assets/icon_TT.png -------------------------------------------------------------------------------- /frontend/src/assets/ig_bet_btn.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thundercore/DoubleOrNothing/dd0501343022f2c4f5217c42a69fad2280306e84/frontend/src/assets/ig_bet_btn.png -------------------------------------------------------------------------------- /frontend/src/assets/ig_coin_card.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thundercore/DoubleOrNothing/dd0501343022f2c4f5217c42a69fad2280306e84/frontend/src/assets/ig_coin_card.png -------------------------------------------------------------------------------- /frontend/src/assets/ig_double_btn.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thundercore/DoubleOrNothing/dd0501343022f2c4f5217c42a69fad2280306e84/frontend/src/assets/ig_double_btn.png -------------------------------------------------------------------------------- /frontend/src/assets/ig_lose_coin.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thundercore/DoubleOrNothing/dd0501343022f2c4f5217c42a69fad2280306e84/frontend/src/assets/ig_lose_coin.png -------------------------------------------------------------------------------- /frontend/src/assets/ig_reset_btn.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thundercore/DoubleOrNothing/dd0501343022f2c4f5217c42a69fad2280306e84/frontend/src/assets/ig_reset_btn.png -------------------------------------------------------------------------------- /frontend/src/assets/ig_win_coin.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thundercore/DoubleOrNothing/dd0501343022f2c4f5217c42a69fad2280306e84/frontend/src/assets/ig_win_coin.png -------------------------------------------------------------------------------- /frontend/src/assets/tt-usdt.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/thundercore/DoubleOrNothing/dd0501343022f2c4f5217c42a69fad2280306e84/frontend/src/assets/tt-usdt.png -------------------------------------------------------------------------------- /frontend/src/components/App/App.css: -------------------------------------------------------------------------------- 1 | html, body, .app, #root { 2 | height: 100%; 3 | font-family: 'Russo One', sans-serif; 4 | } 5 | 6 | .app { 7 | background-image: url("../../assets/bg.jpg"); 8 | background-repeat: no-repeat; 9 | background-size: cover; 10 | background-position: center; 11 | min-height: 510px; 12 | } 13 | 14 | .app-message { 15 | padding-top: 5vh; 16 | text-align: center; 17 | color: white; 18 | } 19 | -------------------------------------------------------------------------------- /frontend/src/components/App/App.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import './App.css' 3 | import { abi } from './DoubleOrNothing.json' 4 | import Web3Loader from '../Web3Loader/Web3Loader' 5 | import ContractLoader from '../ContractLoader/ContractLoader' 6 | import { IWeb3Context } from '../../contexts/Web3Context/Web3Context' 7 | import { IContractContext } from '../../contexts/ContractContext/ContractContext' 8 | import GameContainer from '../../containers/GameContainer/GameContainer' 9 | import config from '../../config' 10 | import { abi as Erc677Abi } from './ERC677.json' 11 | import { ContractEnum } from '../../ContractEnum' 12 | 13 | interface IAppProps {} 14 | interface IAppState {} 15 | 16 | export class App extends React.PureComponent { 17 | renderIncapable() { 18 | return ( 19 |
20 |
Please connect a Web3 provider
21 |
22 | ) 23 | } 24 | 25 | renderPleaseEnable() { 26 | return ( 27 |
28 |
Please enable your Web3 Provider
29 |
30 | ) 31 | } 32 | 33 | renderEnabled = (params: IWeb3Context) => { 34 | return ( 35 |
36 | 50 | this.renderGame(contractParams, params.address) 51 | } 52 | renderLoading={this.renderLoading} 53 | renderUnavailable={this.renderUnavailable} 54 | /> 55 |
56 | ) 57 | } 58 | 59 | renderLoading() { 60 | return
Loading...
61 | } 62 | 63 | renderUnavailable() { 64 | return
Error loading contract
65 | } 66 | 67 | renderGame(params: IContractContext, address: string) { 68 | return ( 69 | 77 | ) 78 | } 79 | 80 | render() { 81 | return ( 82 | 88 | ) 89 | } 90 | } 91 | 92 | export default App 93 | -------------------------------------------------------------------------------- /frontend/src/components/App/DoubleOrNothing.json: -------------------------------------------------------------------------------- 1 | ../../../../smart-contracts/build/contracts/DoubleOrNothing.json -------------------------------------------------------------------------------- /frontend/src/components/Coin/Coin.css: -------------------------------------------------------------------------------- 1 | .coin-container { 2 | transform-style: preserve-3d; 3 | transition: 0.5s all; 4 | position: relative; 5 | width: 140px; 6 | height: 140px; 7 | border-radius: 50%; 8 | } 9 | 10 | .flipped { 11 | transform: rotateX(180deg); 12 | } 13 | 14 | .face { 15 | backface-visibility: hidden; 16 | } 17 | 18 | .top-face { 19 | transform: translateZ(8px); 20 | } 21 | 22 | .top-back-face { 23 | background: #444; 24 | } 25 | 26 | .disk-body { 27 | width: 140px; 28 | height: 16px; 29 | position: absolute; 30 | background: #444; 31 | backface-visibility: hidden; 32 | transform: translate3d(0,63px,0) rotateX(90deg); 33 | } 34 | 35 | .disk-body.back { 36 | background: #938506; 37 | transform: translate3d(0,63px,0) rotateX(90deg) rotateY(180deg); 38 | } 39 | 40 | .disk { 41 | overflow: hidden; 42 | border-radius: 50%; 43 | position: absolute; 44 | height: 100%; 45 | width: 100%; 46 | background: #444; 47 | } 48 | 49 | .bottom-front-face { 50 | background-size: contain; 51 | background-image: url("../../assets/ig_lose_coin.png"); 52 | transform: translateZ(-8px) rotateX(180deg); 53 | box-shadow: 0 0 8px rgba(0, 0, 0, 0.45); 54 | } 55 | 56 | .bottom-back-face { 57 | transform: translateZ(-8px); 58 | background: #444; 59 | } 60 | 61 | .half-face-lower { 62 | top: 50%; 63 | background: #938506; 64 | } 65 | 66 | .half-face-upper { 67 | background: #444; 68 | } 69 | 70 | .half-face { 71 | position: absolute; 72 | height: 50%; 73 | width: 100%; 74 | } 75 | 76 | .top-face-front { 77 | background-size: contain; 78 | background-image: url("../../assets/ig_win_coin.png"); 79 | } 80 | -------------------------------------------------------------------------------- /frontend/src/components/Coin/Coin.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import './Coin.css' 3 | 4 | interface ICoinProps { 5 | showFace: boolean 6 | } 7 | 8 | export class Coin extends React.PureComponent { 9 | render() { 10 | return ( 11 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 | ) 28 | } 29 | } 30 | 31 | export default Coin 32 | -------------------------------------------------------------------------------- /frontend/src/components/CoinPicker/CoinPicker.css: -------------------------------------------------------------------------------- 1 | .coin-picker-container { 2 | display: flex; 3 | width: 100%; 4 | } 5 | 6 | .coin-button-container { 7 | flex: 1; 8 | padding: 6px 12px; 9 | } 10 | 11 | .coin-picker-button-icon { 12 | width: 20px; 13 | height: 20px; 14 | margin-right: 6px; 15 | } 16 | 17 | .coin-picker-button { 18 | width: 100%; 19 | height: 100%; 20 | outline: none; 21 | filter: grayscale(100%); 22 | display: flex; 23 | align-items: center; 24 | justify-content: center; 25 | min-height: 42px; 26 | font-family: inherit; 27 | cursor: pointer; 28 | border-radius: 8px; 29 | color: white; 30 | font-size: 16px; 31 | font-weight: bold; 32 | border: 3px solid rgb(197,197,197); 33 | background-color: rgb(222,222,222); 34 | transition: 0.25s ease all 35 | } 36 | 37 | .coin-picker-button.active { 38 | cursor: default; 39 | filter: grayscale(0%); 40 | border-color: rgb(254,159,37); 41 | background-color: rgb(255,197,41); 42 | } 43 | -------------------------------------------------------------------------------- /frontend/src/components/CoinPicker/CoinPicker.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import './CoinPicker.css' 3 | import TTUsdtImg from '../../assets/tt-usdt.png' 4 | import TTImg from '../../assets/icon_TT.png' 5 | import { ContractEnum } from '../../ContractEnum' 6 | 7 | interface ICoinPickerProps { 8 | active: ContractEnum 9 | changeContract(coin: ContractEnum): any 10 | } 11 | 12 | const buttons = [ 13 | { 14 | name: ContractEnum.DoubleOrNothing, 15 | img: TTImg, 16 | text: 'TT' 17 | }, 18 | { 19 | name: ContractEnum.TTUsdt, 20 | img: TTUsdtImg, 21 | text: 'TT USDT' 22 | }, 23 | ] 24 | 25 | export class CoinPicker extends React.PureComponent { 26 | render() { 27 | const { active, changeContract } = this.props 28 | return ( 29 |
30 | {buttons.map(button => ( 31 |
32 | 46 |
47 | ))} 48 |
49 | ) 50 | } 51 | } 52 | 53 | export default CoinPicker 54 | -------------------------------------------------------------------------------- /frontend/src/components/ContractLoader/ContractLoader.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | 3 | import ContractProvider, { 4 | ContractConsumer, 5 | IContractContext 6 | } from '../../contexts/ContractContext/ContractContext' 7 | import { Signer } from 'ethers' 8 | 9 | interface IContractData { 10 | abi: any[] 11 | address: string 12 | signer: Signer 13 | } 14 | 15 | interface IContractLoaderProps { 16 | contractData: IContractData[] 17 | renderLoading(params: IContractContext): React.ReactNode 18 | renderEnabled(params: IContractContext): React.ReactNode 19 | renderUnavailable(params: IContractContext): React.ReactNode 20 | } 21 | 22 | interface IContractLoaderState { 23 | contracts: IContractData[] 24 | } 25 | 26 | export class ContractLoader extends React.PureComponent< 27 | IContractLoaderProps, 28 | IContractLoaderState 29 | > { 30 | static getDerivedStateFromProps(props: IContractLoaderProps) { 31 | return { 32 | contracts: props.contractData.map(data => ({ 33 | ...data, 34 | checkOnLoad: true 35 | })) 36 | } 37 | } 38 | state: IContractLoaderState = { 39 | contracts: [] 40 | } 41 | 42 | renderContractState = (params: IContractContext) => { 43 | // @ts-ignore 44 | if (Object.keys(params).find(key => params[key].loading)) { 45 | return this.props.renderLoading(params) 46 | // @ts-ignore 47 | } else if (Object.keys(params).find(key => params[key].error)) { 48 | return this.props.renderUnavailable(params) 49 | } 50 | return this.props.renderEnabled(params) 51 | } 52 | 53 | render() { 54 | return ( 55 | 56 | {this.renderContractState} 57 | 58 | ) 59 | } 60 | } 61 | 62 | export default ContractLoader 63 | -------------------------------------------------------------------------------- /frontend/src/components/Display/Display.css: -------------------------------------------------------------------------------- 1 | .display { 2 | display: flex; 3 | flex-direction: row; 4 | flex-wrap: wrap; 5 | justify-content: space-between; 6 | width: 100%; 7 | } 8 | 9 | .title-container { 10 | font-size: 1.25em; 11 | color: #ffffff; 12 | margin: 8px; 13 | display: flex; 14 | flex-direction: row; 15 | } 16 | 17 | .value { 18 | margin-left: 6px; 19 | } 20 | 21 | @media (max-width: 400px) { 22 | .title-container { 23 | font-size: 1em; 24 | } 25 | } 26 | -------------------------------------------------------------------------------- /frontend/src/components/Display/Display.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import './Display.css' 3 | 4 | interface IDisplayProps { 5 | balance: string 6 | betAmount: string 7 | symbol: string 8 | } 9 | 10 | export class Display extends React.PureComponent { 11 | render() { 12 | const { balance, betAmount, symbol } = this.props 13 | return ( 14 |
15 |
16 |
Balance:
17 |
18 | {parseFloat(balance).toFixed(2)} {symbol} 19 |
20 |
21 |
22 |
Bet Size:
23 |
{betAmount} {symbol}
24 |
25 |
26 | ) 27 | } 28 | } 29 | 30 | export default Display 31 | -------------------------------------------------------------------------------- /frontend/src/components/Game/Game.css: -------------------------------------------------------------------------------- 1 | .game-container { 2 | display: flex; 3 | max-width: 420px; 4 | width: 95%; 5 | margin: auto; 6 | align-items: center; 7 | flex-direction: column; 8 | margin-top: 1em; 9 | } 10 | 11 | .display-container { 12 | margin-bottom: 1.5em; 13 | width: 100%; 14 | } 15 | 16 | .coin-display-container { 17 | position: relative; 18 | width: 90%; 19 | } 20 | 21 | .coin-display-container img { 22 | width: 100%; 23 | } 24 | 25 | .flip { 26 | animation: flip 0.5s linear 3; 27 | } 28 | 29 | .container { 30 | text-align: center; 31 | margin-bottom: 1em; 32 | } 33 | 34 | .center { 35 | position: absolute; 36 | top: 50%; 37 | left: 50%; 38 | transform: translate(-50%,-50%) 39 | } 40 | 41 | .coin { 42 | transform-style: preserve-3d; 43 | transform: rotateX(15deg); 44 | transition: all 0.5s linear; 45 | } 46 | 47 | .button { 48 | outline: none; 49 | border: none; 50 | height: 60px; 51 | width: 150px; 52 | background-color: transparent; 53 | background-size: contain; 54 | background-repeat: no-repeat; 55 | transition: all 0.25s ease; 56 | } 57 | 58 | .button[disabled] { 59 | filter: grayscale(1); 60 | } 61 | 62 | .bet-button { 63 | background-image: url("../../assets/ig_bet_btn.png"); 64 | } 65 | 66 | 67 | .reset-button { 68 | background-image: url("../../assets/ig_reset_btn.png"); 69 | } 70 | 71 | .double-button { 72 | background-image: url("../../assets/ig_double_btn.png"); 73 | } 74 | 75 | 76 | @keyframes flip { 77 | 0% { 78 | transform: rotateX(10deg); 79 | } 80 | 100% { 81 | transform: rotateX(370deg); 82 | } 83 | } 84 | 85 | #debug-messages { 86 | background-image: none; 87 | background-color: white; 88 | font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New', monospace; 89 | } 90 | -------------------------------------------------------------------------------- /frontend/src/components/Game/Game.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import './Game.css' 3 | import Coin from '../Coin/Coin' 4 | import Display from '../Display/Display' 5 | import cardSrc from '../../assets/ig_coin_card.png' 6 | import CoinPicker from '../CoinPicker/CoinPicker' 7 | import { ContractEnum } from '../../ContractEnum' 8 | import { formatUnits } from 'ethers/utils' 9 | 10 | interface IGameProps { 11 | active: ContractEnum 12 | changeContract(coin: ContractEnum): any 13 | playGame(): any 14 | double(): any 15 | reset(): any 16 | flipping: boolean 17 | disabled: boolean 18 | win: boolean 19 | balance: string 20 | betAmount: number 21 | decimals: number 22 | symbol: string 23 | } 24 | 25 | interface IGameState { 26 | animating: boolean 27 | } 28 | 29 | export class Game extends React.PureComponent { 30 | static getDerivedStateFromProps(props: IGameProps, state: IGameState) { 31 | return props.flipping && !state.animating ? { animating: true } : null 32 | } 33 | 34 | state = { 35 | animating: false 36 | } 37 | 38 | render() { 39 | const { 40 | playGame, 41 | disabled, 42 | win, 43 | double, 44 | balance, 45 | betAmount, 46 | reset, 47 | active, 48 | changeContract, 49 | decimals, 50 | symbol, 51 | } = this.props 52 | const { animating } = this.state 53 | let b = balance.length === 0 ? '0' : balance 54 | return ( 55 |
56 | 57 |
58 | 59 |
60 |
61 | 62 |
63 |
{ 66 | this.setState({ animating: false }) 67 | }} 68 | > 69 | 70 |
71 |
72 |
73 |
74 |
80 |
81 |
92 |
93 | ) 94 | } 95 | } 96 | 97 | export default Game 98 | -------------------------------------------------------------------------------- /frontend/src/components/Header/Header.css: -------------------------------------------------------------------------------- 1 | .header { 2 | z-index: 2000; 3 | padding: 1em 1.5em; 4 | background: rgb(72,25,124); 5 | color: white; 6 | display: flex; 7 | justify-content: space-between; 8 | align-items: center; 9 | } 10 | 11 | .header-reward { 12 | display: flex; 13 | justify-content: center; 14 | align-items: center; 15 | } 16 | 17 | .reward-container { 18 | border-radius: 5px; 19 | margin: 0 8px; 20 | background: rgb(42,14,74); 21 | padding: 2px 8px; 22 | } 23 | 24 | .arrow-circle { 25 | transition: 0.25s all; 26 | } 27 | 28 | .arrow-circle.open { 29 | transform: rotateX(180deg); 30 | } 31 | -------------------------------------------------------------------------------- /frontend/src/components/Header/Header.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import './Header.css' 3 | import { ReactComponent as ArrowCircle } from '../../assets/ic_arrow-down-circle.svg' 4 | import ReferralRules from '../ReferralRules/ReferralRules' 5 | interface IHeaderProps { 6 | address: string 7 | rewards: string 8 | refereeCount: string 9 | activeTimestamp: number 10 | } 11 | interface IHeaderState { 12 | open: boolean 13 | } 14 | 15 | export class Header extends React.PureComponent { 16 | state = { 17 | open: false 18 | } 19 | 20 | calcualteEther(wei: string) { 21 | const amt = wei.padStart(19, '0') 22 | const decimalIdx = amt.length - 18 23 | const precisionIdx = decimalIdx + 4 24 | // round the next precision 25 | const roundUp = amt.slice(precisionIdx, precisionIdx + 1) >= '5' 26 | let decimal = roundUp 27 | ? amt.slice(decimalIdx, precisionIdx - 1) + 28 | (parseInt(amt.slice(precisionIdx - 1, precisionIdx)) + 1).toString() 29 | : amt.slice(decimalIdx, precisionIdx) 30 | decimal = decimal.replace(/0+$/, '') 31 | return amt.slice(0, decimalIdx) + (decimal ? '.' + decimal : '') 32 | } 33 | 34 | toggle = () => this.setState({ open: !this.state.open }) 35 | 36 | render() { 37 | const { open } = this.state 38 | return ( 39 | <> 40 |
41 |
Double or Nothing
42 |
43 |
Reward
44 |
45 | {this.calcualteEther(this.props.rewards)} 46 |
47 | 51 |
52 |
53 | {open && ( 54 | 60 | )} 61 | 62 | ) 63 | } 64 | } 65 | 66 | export default Header 67 | -------------------------------------------------------------------------------- /frontend/src/components/ReferralRules/ReferralRules.css: -------------------------------------------------------------------------------- 1 | .ReferralRule-container { 2 | top:55px; 3 | bottom:0; 4 | width: 100%; 5 | overflow: auto; 6 | position: absolute; 7 | z-index: 200; 8 | background: rgba(29,6,34,0.9); 9 | } 10 | 11 | .inner-container { 12 | max-width: 420px; 13 | margin: auto; 14 | color: white; 15 | padding: 24px; 16 | } 17 | 18 | .share-container { 19 | line-height: 1.5; 20 | font-size: 1.15em; 21 | text-align: center; 22 | border-radius: 5px; 23 | background: rgba(72,26,86,0.75); 24 | padding: 24px; 25 | margin-bottom: 1.5em; 26 | } 27 | 28 | .referral-title { 29 | text-align: center; 30 | color: rgb(250,245,40); 31 | margin-bottom: 1em; 32 | } 33 | 34 | .referral-link-container { 35 | display: flex; 36 | align-items: stretch; 37 | justify-content: center; 38 | margin-bottom: 1.5em; 39 | } 40 | 41 | .referral-link { 42 | font-size: .85em; 43 | font-family: sans-serif; 44 | word-break: break-all; 45 | padding: 12px; 46 | background: rgba(56,34,63, 0.75); 47 | border-radius: 5px; 48 | flex: 1; 49 | } 50 | 51 | .copy-button { 52 | cursor: pointer; 53 | margin-bottom: 6px; 54 | border: none; 55 | outline: none; 56 | font-size: 1.5em; 57 | font-family: inherit; 58 | padding: 20px; 59 | margin-left: 8px; 60 | border-radius: 5px; 61 | color: black; 62 | background: rgb(255,197,41); 63 | box-shadow: 0 6px 2px rgb(186,117,26); 64 | transition: 0.25s all; 65 | } 66 | 67 | .copy-button.clicked { 68 | margin-top: 6px; 69 | margin-bottom: 0; 70 | box-shadow: 0 0 0 rgb(186,117,26); 71 | } 72 | 73 | .referral-stats-container { 74 | margin-bottom: 2em; 75 | } 76 | 77 | .referral-stat { 78 | display: flex; 79 | justify-content: space-between; 80 | margin-bottom: 0.75em; 81 | font-size: 0.85em; 82 | } 83 | 84 | .referral-rule-block { 85 | margin-bottom: 1em; 86 | } 87 | 88 | .table { 89 | display: flex; 90 | border-radius: 5px; 91 | overflow: hidden; 92 | margin-bottom: 1em; 93 | } 94 | 95 | .table-column { 96 | flex: 1; 97 | } 98 | 99 | .table-column.dark { 100 | background-color: rgb(70,19,81); 101 | } 102 | 103 | .table-column.light { 104 | background-color: rgb(134,52,153); 105 | } 106 | 107 | .table-column.med { 108 | background-color: rgb(93,34,107); 109 | } 110 | 111 | .cell { 112 | display: flex; 113 | padding: 6px; 114 | align-items: center; 115 | justify-content: center; 116 | height: 50px; 117 | text-align: center; 118 | border: 1px solid rgba(255,255,255,0.05); 119 | } 120 | -------------------------------------------------------------------------------- /frontend/src/components/ReferralRules/ReferralRules.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import Countdown, { CountdownRenderProps } from 'react-countdown-now' 3 | 4 | import './ReferralRules.css' 5 | 6 | interface IReferralRuleProps { 7 | address: string 8 | refereeCount: string 9 | reward: string 10 | activeTimestamp: number 11 | } 12 | 13 | interface IReferralRuleState { 14 | clicked: boolean 15 | } 16 | 17 | export class ReferralRules extends React.PureComponent< 18 | IReferralRuleProps, 19 | IReferralRuleState 20 | > { 21 | state = { 22 | clicked: false 23 | } 24 | 25 | renderStat(title: string, val: any) { 26 | return ( 27 |
28 |
{title}
29 |
{val}
30 |
31 | ) 32 | } 33 | 34 | referralLink = () => { 35 | return window.location.origin + `?referrer=${this.props.address}` 36 | } 37 | 38 | copy = () => { 39 | const input = document.createElement('input') 40 | document.body.append(input) 41 | input.value = this.referralLink() 42 | input.select() 43 | document.execCommand('copy') 44 | document.body.removeChild(input) 45 | this.setState({ clicked: true }) 46 | } 47 | 48 | renderColumn(header: string, val: string, className: string) { 49 | return ( 50 |
51 |
{header}
52 |
{val}
53 |
54 | ) 55 | } 56 | 57 | renderCountdown({ 58 | hours, 59 | minutes, 60 | seconds, 61 | completed 62 | }: CountdownRenderProps) { 63 | function paddingTime(val: number) { 64 | return ('00' + val).slice(-2) 65 | } 66 | 67 | return completed ? ( 68 | 00:00:00 69 | ) : ( 70 | 71 | {paddingTime(hours)}:{paddingTime(minutes)}:{paddingTime(seconds)} 72 | 73 | ) 74 | } 75 | 76 | render() { 77 | const { clicked } = this.state 78 | const { refereeCount, reward, activeTimestamp } = this.props 79 | return ( 80 |
81 |
82 |
83 | Share Your Referral Link to Earn More Rewards! 84 |
85 |
Your Referral Link
86 |
87 |
{this.referralLink()}
88 | 94 |
95 |
96 | {this.renderStat('Total Referral Reward', reward)} 97 | {this.renderStat('Number of Referees', refereeCount)} 98 | {this.renderStat( 99 | 'Your referral active time', 100 | 104 | )} 105 |
106 |
107 |
Referral Rule
108 |
109 | For every transaction, 3% of the transaction value is reserved for 110 | the referrers. 60 % of the reserved amount is added to the first 111 | level referrer’s bonus pool, 30% is added to the 2nd level 112 | referrer’s bonus pool, and 10% is added to the 3rd level 113 | referrer’s bonus pool. 114 |
115 |
116 | {this.renderColumn('Bonus Rate', '%', 'dark')} 117 | {this.renderColumn('Level 1', '60 %', 'light')} 118 | {this.renderColumn('Level 2', '30 %', 'med')} 119 | {this.renderColumn('Level 3', '10 %', 'dark')} 120 |
121 | 122 |
123 | As a referrer, you will be awarded 50% of your bonus pool if you 124 | have 4 or less referees, 75% of the pool if you have 5 -24 125 | referees, and 100% if you have 25 or more referees. 126 |
127 | 128 |
129 | {this.renderColumn('Referee Amount', '%', 'dark')} 130 | {this.renderColumn('1-4', '50 %', 'dark')} 131 | {this.renderColumn('5-24', '75 %', 'med')} 132 | {this.renderColumn('25+', '100 %', 'light')} 133 |
134 |
135 |
136 |
137 | ) 138 | } 139 | } 140 | 141 | export default ReferralRules 142 | -------------------------------------------------------------------------------- /frontend/src/components/Web3Loader/Web3Loader.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import Web3Provider, { 3 | IWeb3Context, 4 | Web3Consumer 5 | } from '../../contexts/Web3Context/Web3Context' 6 | 7 | interface IWeb3LoaderProps { 8 | renderWeb3Incapable(params: IWeb3Context): React.ReactNode 9 | renderEnabling(params: IWeb3Context): React.ReactNode 10 | renderEnabled(params: IWeb3Context): React.ReactNode 11 | renderUnabled(params: IWeb3Context): React.ReactNode 12 | } 13 | interface IWeb3LoaderState {} 14 | 15 | export class Web3Loader extends React.PureComponent< 16 | IWeb3LoaderProps, 17 | IWeb3LoaderState 18 | > { 19 | renderAppState = (params: IWeb3Context) => { 20 | if (!params.signer) { 21 | return this.props.renderWeb3Incapable(params) 22 | } else if (params.enabling) { 23 | return this.props.renderEnabling(params) 24 | } else if (params.enabled) { 25 | return this.props.renderEnabled(params) 26 | } 27 | return this.props.renderUnabled(params) 28 | } 29 | 30 | render() { 31 | return ( 32 | 33 | {this.renderAppState} 34 | 35 | ) 36 | } 37 | } 38 | 39 | export default Web3Loader 40 | -------------------------------------------------------------------------------- /frontend/src/config.ts: -------------------------------------------------------------------------------- 1 | import { ContractEnum } from './ContractEnum' 2 | 3 | export default { 4 | [ContractEnum.DoubleOrNothing]: process.env.REACT_APP_DOUBLE_CONTRACT_ADDRESS, 5 | [ContractEnum.TTUsdt]: process.env.REACT_APP_TT_USDT_CONTRACT 6 | } 7 | -------------------------------------------------------------------------------- /frontend/src/containers/DoubleGameContainer/DoubleGameContainer.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import { Contract } from 'ethers' 3 | import { bigNumberify, parseEther } from 'ethers/utils' 4 | import { TransactionReceipt, TransactionResponse } from 'ethers/providers' 5 | import Game from '../../components/Game/Game' 6 | import Header from '../../components/Header/Header' 7 | import { ContractEnum } from '../../ContractEnum' 8 | 9 | interface IDoubleGameContainerProps { 10 | contract: Contract 11 | address: string 12 | changeContract(coin: ContractEnum): any 13 | } 14 | 15 | interface IAccountInfo { 16 | reward: string 17 | referredCount: string 18 | activeTimestamp: number 19 | } 20 | 21 | interface StateWithDebugLogs { 22 | debugMessages: string[] 23 | } 24 | 25 | interface IDoubleGameContainerState extends StateWithDebugLogs{ 26 | referrer?: string 27 | betAmount: number 28 | balance: string 29 | disabled: boolean 30 | flipping: boolean 31 | win: boolean 32 | accountInfo: IAccountInfo 33 | } 34 | 35 | const INITIAL_AMOUNT = 0.5 36 | 37 | export class DoubleGameContainer extends React.PureComponent< 38 | IDoubleGameContainerProps, 39 | IDoubleGameContainerState 40 | > { 41 | intervals: NodeJS.Timeout[] = [] 42 | 43 | constructor(props: IDoubleGameContainerProps) { 44 | super(props) 45 | this.watchBalance() 46 | this.watchReferrals() 47 | const referrerMatch = window.location.search.match( 48 | /referrer=(0x[a-fA-F0-9]{40})/ 49 | ) 50 | const address = referrerMatch ? referrerMatch[1] : '' 51 | this.state = { 52 | referrer: address.toLowerCase(), 53 | betAmount: INITIAL_AMOUNT, 54 | balance: '', 55 | win: true, 56 | disabled: false, 57 | flipping: false, 58 | accountInfo: { 59 | reward: '0', 60 | referredCount: '0', 61 | activeTimestamp: Date.now() 62 | }, 63 | debugMessages: [], 64 | } 65 | } 66 | 67 | componentWillUnmount(): void { 68 | this.intervals.forEach(interval => clearInterval(interval)) 69 | } 70 | 71 | watchBalance = () => { 72 | this.intervals.push( 73 | setInterval(() => { 74 | this.props.contract.provider 75 | .getBalance(this.props.address) 76 | .then(bal => { 77 | this.setState({ balance: bal.toString() }) 78 | }) 79 | }, 500) 80 | ) 81 | } 82 | 83 | watchReferrals = () => { 84 | this.intervals.push( 85 | setInterval(() => { 86 | this.props.contract.accounts(this.props.address).then((info: any) => { 87 | this.setState({ 88 | accountInfo: { 89 | reward: info.reward.toString(), 90 | referredCount: info.referredCount.toString(), 91 | activeTimestamp: 92 | info.lastActiveTimestamp.toString() * 1000 + 24 * 60 * 60 * 1000 93 | } 94 | }) 95 | }) 96 | }, 500) 97 | ) 98 | } 99 | 100 | log = (v: any) => { 101 | this.setState(prevState => ({ 102 | debugMessages: [`${v}`].concat(prevState.debugMessages) 103 | })) 104 | } 105 | 106 | play = (val: number) => { 107 | this.setState({ disabled: true }) 108 | const params: any = [ 109 | { 110 | value: parseEther(val.toString()).toHexString(), 111 | // because there is an if else based on time, the estimate gas will fail and the transaction will fail 112 | // you must set the gas limit 113 | gasPrice: bigNumberify('1000000000').toHexString(), 114 | gasLimit: bigNumberify('2000000').toHexString() 115 | } 116 | ] 117 | const referrer = !!this.state.referrer 118 | if (referrer) { 119 | params.unshift(this.state.referrer) 120 | } 121 | this.props.contract[referrer ? 'bet(address)' : 'bet()'](...params) 122 | .then((trans: TransactionResponse) => { 123 | this.setState({ flipping: true }) 124 | return trans.wait() 125 | }) 126 | .then((receipt: TransactionReceipt) => { 127 | //@ts-ignore 128 | const event = receipt.events.find(evt => evt.event === 'BetSettled') 129 | 130 | this.setState({ 131 | flipping: false, 132 | win: event.args.winnings.toString() !== '0' 133 | }) 134 | }) 135 | .catch((err: Error) => { 136 | console.log(`setState("logs", ${err})`) 137 | this.log(JSON.stringify(err)) 138 | }) 139 | .finally(() => { 140 | this.setState({ disabled: false, flipping: false }) 141 | }) 142 | } 143 | 144 | playGame = () => { 145 | this.play(this.state.betAmount) 146 | } 147 | 148 | double = () => { 149 | const newBet = this.state.betAmount * 2 150 | this.setState({ betAmount: newBet }) 151 | this.play(newBet) 152 | } 153 | 154 | reset = () => { 155 | this.setState({ betAmount: INITIAL_AMOUNT }) 156 | } 157 | 158 | render() { 159 | const { 160 | disabled, 161 | flipping, 162 | win, 163 | balance, 164 | betAmount, 165 | accountInfo 166 | } = this.state 167 | return ( 168 | <> 169 |
175 | 189 |
190 | {this.state.debugMessages.map((message, index) => (
{message}
))} 191 |
192 | 193 | ) 194 | } 195 | } 196 | 197 | export default DoubleGameContainer 198 | -------------------------------------------------------------------------------- /frontend/src/containers/Erc677GameContainer/Erc677GameContainer.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import { Contract } from 'ethers' 3 | import { BigNumber, bigNumberify, parseUnits } from 'ethers/utils' 4 | import Game from '../../components/Game/Game' 5 | import Header from '../../components/Header/Header' 6 | import { ContractEnum } from '../../ContractEnum' 7 | import { TransactionReceipt, TransactionResponse } from 'ethers/providers' 8 | 9 | interface IErc677GameContainerProps { 10 | contract: Contract 11 | address: string 12 | gameAddress: string 13 | contractName: ContractEnum 14 | changeContract(coin: ContractEnum): any 15 | } 16 | 17 | interface IAccountInfo { 18 | reward: string 19 | referredCount: string 20 | activeTimestamp: number 21 | } 22 | 23 | interface IErc677GameContainerState { 24 | betAmount: number 25 | balance: string 26 | disabled: boolean 27 | flipping: boolean 28 | win: boolean 29 | accountInfo: IAccountInfo 30 | decimals: number 31 | } 32 | 33 | const INITIAL_AMOUNT = 0.5 34 | 35 | export class Erc677GameContainer extends React.PureComponent< 36 | IErc677GameContainerProps, 37 | IErc677GameContainerState 38 | > { 39 | interval?: NodeJS.Timeout 40 | 41 | constructor(props: IErc677GameContainerProps) { 42 | super(props) 43 | this.state = { 44 | betAmount: INITIAL_AMOUNT, 45 | balance: '', 46 | win: true, 47 | disabled: false, 48 | flipping: false, 49 | accountInfo: { 50 | reward: '0', 51 | referredCount: '0', 52 | activeTimestamp: Date.now() 53 | }, 54 | decimals: 6 55 | } 56 | this.watchBalance() 57 | } 58 | 59 | componentWillUnmount(): void { 60 | if (this.interval) { 61 | clearInterval(this.interval) 62 | } 63 | } 64 | 65 | watchBalance = () => { 66 | this.interval = setInterval(() => { 67 | this.props.contract 68 | .balanceOf(this.props.address) 69 | .then((bal: BigNumber) => this.props.contract.decimals() 70 | .then((d: number) => { 71 | this.setState({ balance: bal.toString(), decimals: d }) 72 | } 73 | )) 74 | }, 500) 75 | } 76 | 77 | play = (val: number) => { 78 | this.setState({ disabled: true }) 79 | 80 | this.props.contract.decimals().then((d:number) => { 81 | console.log(d) 82 | this.props.contract 83 | .transferAndCall( 84 | this.props.gameAddress, 85 | parseUnits(val.toString(), d), 86 | '0xfae', 87 | { 88 | gasPrice: bigNumberify('1000000000').toHexString(), 89 | gasLimit: bigNumberify('100000').toHexString() 90 | } 91 | ) 92 | .then((trans: TransactionResponse) => { 93 | this.setState({ flipping: true }) 94 | return trans.wait() 95 | }) 96 | .then((receipt: TransactionReceipt) => { 97 | //@ts-ignore 98 | const didWin = !!receipt.events.find( 99 | //@ts-ignore 100 | evt => 101 | evt.event === 'Transfer' && 102 | evt.args.to.toLowerCase() === this.props.address.toLowerCase() 103 | ) 104 | 105 | this.setState({ 106 | flipping: false, 107 | win: didWin 108 | }) 109 | }) 110 | .finally(() => { 111 | this.setState({ disabled: false, flipping: false }) 112 | }) 113 | }) 114 | } 115 | 116 | playGame = () => { 117 | this.play(this.state.betAmount) 118 | } 119 | 120 | double = () => { 121 | const newBet = this.state.betAmount * 2 122 | this.setState({ betAmount: newBet }) 123 | this.play(newBet) 124 | } 125 | 126 | reset = () => { 127 | this.setState({ betAmount: INITIAL_AMOUNT }) 128 | } 129 | 130 | render() { 131 | const { 132 | disabled, 133 | flipping, 134 | win, 135 | balance, 136 | betAmount, 137 | accountInfo, 138 | decimals 139 | } = this.state 140 | return ( 141 | <> 142 |
148 | 162 | 163 | ) 164 | } 165 | } 166 | 167 | export default Erc677GameContainer 168 | -------------------------------------------------------------------------------- /frontend/src/containers/GameContainer/GameContainer.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import { Contract } from 'ethers' 3 | import { ContractEnum } from '../../ContractEnum' 4 | import DoubleGameContainer from '../DoubleGameContainer/DoubleGameContainer' 5 | import Erc677GameContainer from '../Erc677GameContainer/Erc677GameContainer' 6 | 7 | interface IGameContainerProps { 8 | contracts: { 9 | [ContractEnum.DoubleOrNothing]: Contract 10 | [ContractEnum.TTUsdt]: Contract 11 | } 12 | address: string 13 | } 14 | 15 | interface IGameContainerState { 16 | currentContract: ContractEnum 17 | } 18 | 19 | export class GameContainer extends React.PureComponent< 20 | IGameContainerProps, 21 | IGameContainerState 22 | > { 23 | state = { 24 | currentContract: ContractEnum.DoubleOrNothing, 25 | } 26 | 27 | changeContract = (contract: ContractEnum) => { 28 | this.setState({ currentContract: contract }) 29 | } 30 | 31 | render() { 32 | const { currentContract } = this.state 33 | const { address, contracts } = this.props 34 | if (currentContract === ContractEnum.DoubleOrNothing) { 35 | return ( 36 | 41 | ) 42 | } else { 43 | return ( 44 | 51 | ) 52 | 53 | } 54 | } 55 | } 56 | 57 | export default GameContainer 58 | -------------------------------------------------------------------------------- /frontend/src/contexts/ContractContext/ContractContext.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import { Contract, Signer } from 'ethers' 3 | 4 | interface IContractContextProps { 5 | contracts: IContractData[] 6 | } 7 | 8 | interface IContractData { 9 | signer: Signer 10 | checkOnLoad?: boolean 11 | address: string 12 | abi: any[] 13 | } 14 | 15 | interface IContractState { 16 | contract: Contract 17 | loading: boolean 18 | error: boolean 19 | load(): void 20 | } 21 | 22 | interface IContractContextState { 23 | [key: string]: IContractState 24 | } 25 | 26 | export type IContractContext = IContractContextState 27 | 28 | const { Consumer, Provider } = React.createContext({}) 29 | 30 | export class ContractProvider extends React.PureComponent< 31 | IContractContextProps, 32 | IContractContextState 33 | > { 34 | constructor(props: IContractContextProps) { 35 | super(props) 36 | const contractInfo = props.contracts.reduce( 37 | (agg, contractData) => { 38 | agg[contractData.address] = { 39 | contract: new Contract( 40 | contractData.address, 41 | contractData.abi, 42 | contractData.signer 43 | ), 44 | loading: !!contractData.checkOnLoad, 45 | error: false, 46 | load: () => { 47 | this.loadContract(contractData.address) 48 | } 49 | } 50 | return agg 51 | }, 52 | {} as { 53 | [k: string]: IContractState 54 | } 55 | ) 56 | this.state = contractInfo 57 | 58 | props.contracts.forEach(contractData => { 59 | if (contractData.checkOnLoad) { 60 | this.loadContract(contractData.address, true) 61 | } 62 | }) 63 | } 64 | 65 | private loadContract = (address: string, initializing?: boolean) => { 66 | const contractData = this.state[address] 67 | if (!contractData) { 68 | return 69 | } 70 | if (!initializing) { 71 | this.setState({ 72 | [address]: { ...contractData, loading: true, error: false } 73 | }) 74 | } 75 | contractData.contract 76 | .deployed() 77 | .then(() => { 78 | this.setState({ 79 | [address]: { ...contractData, loading: false } 80 | }) 81 | }) 82 | .catch(() => { 83 | this.setState({ 84 | [address]: { ...contractData, loading: false, error: true } 85 | }) 86 | }) 87 | } 88 | 89 | render() { 90 | return {this.props.children} 91 | } 92 | } 93 | 94 | export default ContractProvider 95 | 96 | export const ContractConsumer = Consumer 97 | -------------------------------------------------------------------------------- /frontend/src/contexts/Web3Context/Web3Context.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import { Signer } from 'ethers' 3 | import { Web3Provider as EthersProvider } from 'ethers/providers' 4 | 5 | interface IWeb3ContextProps { 6 | enableOnLoad?: boolean 7 | } 8 | 9 | interface IWeb3ContextState { 10 | signer?: Signer 11 | enabled: boolean 12 | enabling: boolean 13 | address: string 14 | } 15 | 16 | export interface IWeb3Context { 17 | enable(): void 18 | signer?: Signer 19 | enabled: boolean 20 | enabling: boolean 21 | address: string 22 | } 23 | 24 | const { Consumer, Provider } = React.createContext({ 25 | enable: () => {}, 26 | address: '', 27 | signer: undefined, 28 | enabling: false, 29 | enabled: false 30 | }) 31 | 32 | export class Web3Provider extends React.PureComponent< 33 | IWeb3ContextProps, 34 | IWeb3ContextState 35 | > { 36 | constructor(props: IWeb3ContextProps) { 37 | super(props) 38 | const web3Capable = !!window.ethereum 39 | this.state = { 40 | address: '', 41 | signer: web3Capable && new EthersProvider(window.ethereum).getSigner(), 42 | enabled: false, 43 | enabling: web3Capable && !!props.enableOnLoad 44 | } 45 | } 46 | 47 | componentDidMount(): void { 48 | if (this.props.enableOnLoad) { 49 | this.enable() 50 | } 51 | } 52 | 53 | enable = () => { 54 | if (this.state.signer) { 55 | this.setState({ enabling: true }) 56 | this.state.signer 57 | //@ts-ignore 58 | .provider!._web3Provider.enable() 59 | .then(([address]: [string]) => { 60 | this.setState({ enabling: false, enabled: true, address }) 61 | }) 62 | .catch(() => { 63 | this.setState({ 64 | enabling: false, 65 | enabled: false 66 | }) 67 | }) 68 | } 69 | } 70 | 71 | render() { 72 | return ( 73 | 74 | {this.props.children} 75 | 76 | ) 77 | } 78 | } 79 | 80 | export default Web3Provider 81 | 82 | export const Web3Consumer = Consumer 83 | -------------------------------------------------------------------------------- /frontend/src/index.css: -------------------------------------------------------------------------------- 1 | body { 2 | margin: 0; 3 | font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', 'Oxygen', 4 | 'Ubuntu', 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue', 5 | sans-serif; 6 | -webkit-font-smoothing: antialiased; 7 | -moz-osx-font-smoothing: grayscale; 8 | } 9 | 10 | code { 11 | font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New', 12 | monospace; 13 | } 14 | -------------------------------------------------------------------------------- /frontend/src/index.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import ReactDOM from 'react-dom' 3 | import './index.css' 4 | import App from './components/App/App' 5 | import * as serviceWorker from './serviceWorker' 6 | 7 | ReactDOM.render(, document.getElementById('root')) 8 | 9 | // If you want your app to work offline and load faster, you can change 10 | // unregister() to register() below. Note this comes with some pitfalls. 11 | // Learn more about service workers: https://bit.ly/CRA-PWA 12 | serviceWorker.unregister() 13 | -------------------------------------------------------------------------------- /frontend/src/react-app-env.d.ts: -------------------------------------------------------------------------------- 1 | /// 2 | -------------------------------------------------------------------------------- /frontend/src/serviceWorker.ts: -------------------------------------------------------------------------------- 1 | // This optional code is used to register a service worker. 2 | // register() is not called by default. 3 | 4 | // This lets the app load faster on subsequent visits in production, and gives 5 | // it offline capabilities. However, it also means that developers (and users) 6 | // will only see deployed updates on subsequent visits to a page, after all the 7 | // existing tabs open on the page have been closed, since previously cached 8 | // resources are updated in the background. 9 | 10 | // To learn more about the benefits of this model and instructions on how to 11 | // opt-in, read https://bit.ly/CRA-PWA 12 | 13 | const isLocalhost = Boolean( 14 | window.location.hostname === 'localhost' || 15 | // [::1] is the IPv6 localhost address. 16 | window.location.hostname === '[::1]' || 17 | // 127.0.0.1/8 is considered localhost for IPv4. 18 | window.location.hostname.match( 19 | /^127(?:\.(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)){3}$/ 20 | ) 21 | ) 22 | 23 | type Config = { 24 | onSuccess?: (registration: ServiceWorkerRegistration) => void 25 | onUpdate?: (registration: ServiceWorkerRegistration) => void 26 | } 27 | 28 | export function register(config?: Config) { 29 | if (process.env.NODE_ENV === 'production' && 'serviceWorker' in navigator) { 30 | // The URL constructor is available in all browsers that support SW. 31 | const publicUrl = new URL( 32 | (process as { env: { [key: string]: string } }).env.PUBLIC_URL, 33 | window.location.href 34 | ) 35 | if (publicUrl.origin !== window.location.origin) { 36 | // Our service worker won't work if PUBLIC_URL is on a different origin 37 | // from what our page is served on. This might happen if a CDN is used to 38 | // serve assets; see https://github.com/facebook/create-react-app/issues/2374 39 | return 40 | } 41 | 42 | window.addEventListener('load', () => { 43 | const swUrl = `${process.env.PUBLIC_URL}/service-worker.js` 44 | 45 | if (isLocalhost) { 46 | // This is running on localhost. Let's check if a service worker still exists or not. 47 | checkValidServiceWorker(swUrl, config) 48 | 49 | // Add some additional logging to localhost, pointing developers to the 50 | // service worker/PWA documentation. 51 | navigator.serviceWorker.ready.then(() => { 52 | console.log( 53 | 'This web app is being served cache-first by a service ' + 54 | 'worker. To learn more, visit https://bit.ly/CRA-PWA' 55 | ) 56 | }) 57 | } else { 58 | // Is not localhost. Just register service worker 59 | registerValidSW(swUrl, config) 60 | } 61 | }) 62 | } 63 | } 64 | 65 | function registerValidSW(swUrl: string, config?: Config) { 66 | navigator.serviceWorker 67 | .register(swUrl) 68 | .then(registration => { 69 | registration.onupdatefound = () => { 70 | const installingWorker = registration.installing 71 | if (installingWorker == null) { 72 | return 73 | } 74 | installingWorker.onstatechange = () => { 75 | if (installingWorker.state === 'installed') { 76 | if (navigator.serviceWorker.controller) { 77 | // At this point, the updated precached content has been fetched, 78 | // but the previous service worker will still serve the older 79 | // content until all client tabs are closed. 80 | console.log( 81 | 'New content is available and will be used when all ' + 82 | 'tabs for this page are closed. See https://bit.ly/CRA-PWA.' 83 | ) 84 | 85 | // Execute callback 86 | if (config && config.onUpdate) { 87 | config.onUpdate(registration) 88 | } 89 | } else { 90 | // At this point, everything has been precached. 91 | // It's the perfect time to display a 92 | // "Content is cached for offline use." message. 93 | console.log('Content is cached for offline use.') 94 | 95 | // Execute callback 96 | if (config && config.onSuccess) { 97 | config.onSuccess(registration) 98 | } 99 | } 100 | } 101 | } 102 | } 103 | }) 104 | .catch(error => { 105 | console.error('Error during service worker registration:', error) 106 | }) 107 | } 108 | 109 | function checkValidServiceWorker(swUrl: string, config?: Config) { 110 | // Check if the service worker can be found. If it can't reload the page. 111 | fetch(swUrl) 112 | .then(response => { 113 | // Ensure service worker exists, and that we really are getting a JS file. 114 | const contentType = response.headers.get('content-type') 115 | if ( 116 | response.status === 404 || 117 | (contentType != null && contentType.indexOf('javascript') === -1) 118 | ) { 119 | // No service worker found. Probably a different app. Reload the page. 120 | navigator.serviceWorker.ready.then(registration => { 121 | registration.unregister().then(() => { 122 | window.location.reload() 123 | }) 124 | }) 125 | } else { 126 | // Service worker found. Proceed as normal. 127 | registerValidSW(swUrl, config) 128 | } 129 | }) 130 | .catch(() => { 131 | console.log( 132 | 'No internet connection found. App is running in offline mode.' 133 | ) 134 | }) 135 | } 136 | 137 | export function unregister() { 138 | if ('serviceWorker' in navigator) { 139 | navigator.serviceWorker.ready.then(registration => { 140 | registration.unregister() 141 | }) 142 | } 143 | } 144 | -------------------------------------------------------------------------------- /frontend/tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "target": "es5", 4 | "lib": [ 5 | "dom", 6 | "dom.iterable", 7 | "esnext" 8 | ], 9 | "allowJs": true, 10 | "skipLibCheck": true, 11 | "esModuleInterop": true, 12 | "allowSyntheticDefaultImports": true, 13 | "strict": true, 14 | "forceConsistentCasingInFileNames": true, 15 | "module": "esnext", 16 | "moduleResolution": "node", 17 | "resolveJsonModule": true, 18 | "isolatedModules": true, 19 | "noEmit": true, 20 | "jsx": "preserve" 21 | }, 22 | "include": [ 23 | "src", 24 | "typings" 25 | ] 26 | } 27 | -------------------------------------------------------------------------------- /frontend/typings/abi.d.ts: -------------------------------------------------------------------------------- 1 | declare module '*.json' { 2 | export const abi: Abi 3 | } 4 | 5 | declare type Abi = Array<{ 6 | constant: any 7 | inputs: any 8 | name: any 9 | outputs: any 10 | payable: any 11 | stateMutability: any 12 | type: any 13 | }> 14 | -------------------------------------------------------------------------------- /frontend/typings/env.d.ts: -------------------------------------------------------------------------------- 1 | declare namespace NodeJS { 2 | export interface ProcessEnv { 3 | REACT_APP_RPC_URL: string 4 | REACT_APP_DOUBLE_CONTRACT_ADDRESS: string 5 | REACT_APP_TT_USDT_CONTRACT: string 6 | } 7 | } 8 | -------------------------------------------------------------------------------- /frontend/typings/images.d.ts: -------------------------------------------------------------------------------- 1 | declare module '*.svg' 2 | declare module '*.png' 3 | declare module '*.jpg' 4 | declare module '*.gif' 5 | -------------------------------------------------------------------------------- /frontend/typings/sounds.d.ts: -------------------------------------------------------------------------------- 1 | declare module '*.wav' 2 | declare module '*.ogg' 3 | declare module '*.mp3' 4 | -------------------------------------------------------------------------------- /frontend/typings/web3.d.ts: -------------------------------------------------------------------------------- 1 | interface IWeb3Provider { 2 | sendAsync(payload: any, callback: (e: Error, val: any) => void): any 3 | enable(): Promise 4 | } 5 | 6 | interface Window { 7 | ethereum: IWeb3Provider 8 | } 9 | -------------------------------------------------------------------------------- /index.html: -------------------------------------------------------------------------------- 1 | ./frontend/build/index.html -------------------------------------------------------------------------------- /smart-contracts/.gitignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | conda-env 3 | .private-keys 4 | .mnemonic 5 | local.jsonc 6 | -------------------------------------------------------------------------------- /smart-contracts/GNUmakefile: -------------------------------------------------------------------------------- 1 | CONDA := $(HOME)/miniconda3/bin/conda 2 | 3 | TOP_DIR := $(patsubst %/,%,$(dir $(realpath $(lastword $(MAKEFILE_LIST))))) 4 | CONDA_ENV_DIR := $(TOP_DIR)/conda-env 5 | 6 | # This can be used to customized local builds. 7 | -include $(TOP_DIR)/Local.mk 8 | 9 | PATH := $(CONDA_ENV_DIR)/bin:$(TOP_DIR)/node_modules/.bin:$(PATH) 10 | export PATH 11 | 12 | PREFERRED_INTERACTIVE_SHELL ?= bash 13 | PS1_NAME ?= 'double-or-nothing' 14 | MAKE_SHELL_PS1 ?= '$(PS1_NAME) $$ ' 15 | .PHONY: shell 16 | ifeq ($(PREFERRED_INTERACTIVE_SHELL),bash) 17 | shell: 18 | @INIT_FILE=$(shell mktemp); \ 19 | printf '[ -e $$HOME/.bashrc ] && source $$HOME/.bashrc\n' > $$INIT_FILE; \ 20 | printf '[ -e Local.env ] && source Local.env\n' >> $$INIT_FILE; \ 21 | printf 'PS1='"$(MAKE_SHELL_PS1) "'\n' >> $$INIT_FILE; \ 22 | $(PREFERRED_INTERACTIVE_SHELL) --init-file $$INIT_FILE || true 23 | else ifeq ($(PREFERRED_INTERACTIVE_SHELL),fish) 24 | shell: 25 | @INIT_FILE=$(shell mktemp); \ 26 | printf 'if functions -q fish_right_prompt\n' > $$INIT_FILE; \ 27 | printf ' functions -c fish_right_prompt __fish_right_prompt_original\n' >> $$INIT_FILE; \ 28 | printf ' functions -e fish_right_prompt\n' >> $$INIT_FILE; \ 29 | printf 'else\n' >> $$INIT_FILE; \ 30 | printf ' function __fish_right_prompt_original\n' >> $$INIT_FILE; \ 31 | printf ' end\n' >> $$INIT_FILE; \ 32 | printf 'end\n' >> $$INIT_FILE; \ 33 | printf 'function fish_right_prompt\n' >> $$INIT_FILE; \ 34 | printf ' echo -n "($(PS1_NAME)) "\n' >> $$INIT_FILE; \ 35 | printf ' __fish_right_prompt_original\n' >> $$INIT_FILE; \ 36 | printf 'end\n' >> $$INIT_FILE; \ 37 | $(PREFERRED_INTERACTIVE_SHELL) --init-command="source $$INIT_FILE" || true 38 | else 39 | shell: 40 | @$(PREFERRED_INTERACTIVE_SHELL) || true 41 | endif 42 | 43 | .PHONY: conda-env 44 | conda-env: 45 | if [ ! -d $(CONDA_ENV_DIR) ]; then \ 46 | $(CONDA) env create -f environment.yml -p $(CONDA_ENV_DIR); \ 47 | else \ 48 | $(CONDA) env update -f environment.yml -p $(CONDA_ENV_DIR); \ 49 | fi 50 | yarn install 51 | -------------------------------------------------------------------------------- /smart-contracts/build/contracts/ThunderRandomLibraryInterface.json: -------------------------------------------------------------------------------- 1 | { 2 | "contractName": "ThunderRandomLibraryInterface", 3 | "abi": [ 4 | { 5 | "constant": false, 6 | "inputs": [], 7 | "name": "rand", 8 | "outputs": [ 9 | { 10 | "name": "", 11 | "type": "uint256" 12 | } 13 | ], 14 | "payable": false, 15 | "stateMutability": "nonpayable", 16 | "type": "function" 17 | } 18 | ], 19 | "metadata": "", 20 | "bytecode": "0x", 21 | "deployedBytecode": "0x", 22 | "sourceMap": "", 23 | "deployedSourceMap": "", 24 | "source": "contract ThunderRandomLibraryInterface {\n function rand() public returns (uint256);\n}\n", 25 | "sourcePath": "/home/scottt/git/tt/DoubleOrNothing/smart-contracts/contracts/ThunderRandomLibraryInterface.sol", 26 | "ast": { 27 | "absolutePath": "/home/scottt/git/tt/DoubleOrNothing/smart-contracts/contracts/ThunderRandomLibraryInterface.sol", 28 | "exportedSymbols": { 29 | "ThunderRandomLibraryInterface": [ 30 | 666 31 | ] 32 | }, 33 | "id": 667, 34 | "nodeType": "SourceUnit", 35 | "nodes": [ 36 | { 37 | "baseContracts": [], 38 | "contractDependencies": [], 39 | "contractKind": "contract", 40 | "documentation": null, 41 | "fullyImplemented": false, 42 | "id": 666, 43 | "linearizedBaseContracts": [ 44 | 666 45 | ], 46 | "name": "ThunderRandomLibraryInterface", 47 | "nodeType": "ContractDefinition", 48 | "nodes": [ 49 | { 50 | "body": null, 51 | "documentation": null, 52 | "id": 665, 53 | "implemented": false, 54 | "kind": "function", 55 | "modifiers": [], 56 | "name": "rand", 57 | "nodeType": "FunctionDefinition", 58 | "parameters": { 59 | "id": 661, 60 | "nodeType": "ParameterList", 61 | "parameters": [], 62 | "src": "58:2:3" 63 | }, 64 | "returnParameters": { 65 | "id": 664, 66 | "nodeType": "ParameterList", 67 | "parameters": [ 68 | { 69 | "constant": false, 70 | "id": 663, 71 | "name": "", 72 | "nodeType": "VariableDeclaration", 73 | "scope": 665, 74 | "src": "77:7:3", 75 | "stateVariable": false, 76 | "storageLocation": "default", 77 | "typeDescriptions": { 78 | "typeIdentifier": "t_uint256", 79 | "typeString": "uint256" 80 | }, 81 | "typeName": { 82 | "id": 662, 83 | "name": "uint256", 84 | "nodeType": "ElementaryTypeName", 85 | "src": "77:7:3", 86 | "typeDescriptions": { 87 | "typeIdentifier": "t_uint256", 88 | "typeString": "uint256" 89 | } 90 | }, 91 | "value": null, 92 | "visibility": "internal" 93 | } 94 | ], 95 | "src": "76:9:3" 96 | }, 97 | "scope": 666, 98 | "src": "45:41:3", 99 | "stateMutability": "nonpayable", 100 | "superFunction": null, 101 | "visibility": "public" 102 | } 103 | ], 104 | "scope": 667, 105 | "src": "0:88:3" 106 | } 107 | ], 108 | "src": "0:89:3" 109 | }, 110 | "legacyAST": { 111 | "absolutePath": "/home/scottt/git/tt/DoubleOrNothing/smart-contracts/contracts/ThunderRandomLibraryInterface.sol", 112 | "exportedSymbols": { 113 | "ThunderRandomLibraryInterface": [ 114 | 666 115 | ] 116 | }, 117 | "id": 667, 118 | "nodeType": "SourceUnit", 119 | "nodes": [ 120 | { 121 | "baseContracts": [], 122 | "contractDependencies": [], 123 | "contractKind": "contract", 124 | "documentation": null, 125 | "fullyImplemented": false, 126 | "id": 666, 127 | "linearizedBaseContracts": [ 128 | 666 129 | ], 130 | "name": "ThunderRandomLibraryInterface", 131 | "nodeType": "ContractDefinition", 132 | "nodes": [ 133 | { 134 | "body": null, 135 | "documentation": null, 136 | "id": 665, 137 | "implemented": false, 138 | "kind": "function", 139 | "modifiers": [], 140 | "name": "rand", 141 | "nodeType": "FunctionDefinition", 142 | "parameters": { 143 | "id": 661, 144 | "nodeType": "ParameterList", 145 | "parameters": [], 146 | "src": "58:2:3" 147 | }, 148 | "returnParameters": { 149 | "id": 664, 150 | "nodeType": "ParameterList", 151 | "parameters": [ 152 | { 153 | "constant": false, 154 | "id": 663, 155 | "name": "", 156 | "nodeType": "VariableDeclaration", 157 | "scope": 665, 158 | "src": "77:7:3", 159 | "stateVariable": false, 160 | "storageLocation": "default", 161 | "typeDescriptions": { 162 | "typeIdentifier": "t_uint256", 163 | "typeString": "uint256" 164 | }, 165 | "typeName": { 166 | "id": 662, 167 | "name": "uint256", 168 | "nodeType": "ElementaryTypeName", 169 | "src": "77:7:3", 170 | "typeDescriptions": { 171 | "typeIdentifier": "t_uint256", 172 | "typeString": "uint256" 173 | } 174 | }, 175 | "value": null, 176 | "visibility": "internal" 177 | } 178 | ], 179 | "src": "76:9:3" 180 | }, 181 | "scope": 666, 182 | "src": "45:41:3", 183 | "stateMutability": "nonpayable", 184 | "superFunction": null, 185 | "visibility": "public" 186 | } 187 | ], 188 | "scope": 667, 189 | "src": "0:88:3" 190 | } 191 | ], 192 | "src": "0:89:3" 193 | }, 194 | "compiler": { 195 | "name": "solc", 196 | "version": "0.5.0+commit.1d4f565a.Emscripten.clang" 197 | }, 198 | "networks": {}, 199 | "schemaVersion": "3.1.0", 200 | "updatedAt": "2020-05-06T12:23:04.663Z", 201 | "devdoc": { 202 | "methods": {} 203 | }, 204 | "userdoc": { 205 | "methods": {} 206 | } 207 | } -------------------------------------------------------------------------------- /smart-contracts/contracts/DoubleOrNothing.sol: -------------------------------------------------------------------------------- 1 | pragma solidity 0.5.0; 2 | 3 | import '@thundercore/referral-solidity/contracts/Referral.sol'; 4 | import "openzeppelin-solidity/contracts/ownership/Ownable.sol"; 5 | import "openzeppelin-solidity/contracts/token/ERC20/ERC20.sol"; 6 | import "./ERC677Interface.sol"; 7 | import "./ThunderRandomLibraryInterface.sol"; 8 | 9 | contract DoubleOrNothing is Ownable, Referral, ERC677Receiver { 10 | 11 | TokenData[] internal tokens; 12 | ThunderRandomLibraryInterface internal RNGLibrary; 13 | struct TokenData { 14 | IBurnableMintableERC677Token instance; 15 | uint256 lastBalance; // Don't trust this; see onTokenTransfer 16 | } 17 | 18 | event BetSettled(address player, uint256 winnings); 19 | 20 | constructor( 21 | uint _decimals, 22 | uint _referralBonus, 23 | uint _secondsUntilInactive, 24 | bool _onlyRewardActiveReferrers, 25 | uint256[] memory _levelRate, 26 | uint256[] memory _refereeBonusRateMap, 27 | address[] memory _tokenAddresses, 28 | address _randomNumberContractAddress 29 | ) Referral( 30 | _decimals, 31 | _referralBonus, 32 | _secondsUntilInactive, 33 | _onlyRewardActiveReferrers, 34 | _levelRate, 35 | _refereeBonusRateMap 36 | ) public { 37 | for (uint i = 0; i < _tokenAddresses.length; i++) { 38 | tokens.push(generateTokenData(_tokenAddresses[i])); 39 | } 40 | RNGLibrary = ThunderRandomLibraryInterface(_randomNumberContractAddress); 41 | } 42 | 43 | function generateTokenData(address _tokenAddress) internal view returns (TokenData memory) { 44 | IBurnableMintableERC677Token token = IBurnableMintableERC677Token(_tokenAddress); 45 | return TokenData(token, token.balanceOf(address(this))); 46 | } 47 | 48 | // Return a TokenData value passed by reference so we can modify it directly. 49 | function getToken(address _tokenAddress) internal view returns (TokenData storage) { 50 | for (uint i = 0; i < tokens.length; i++) { 51 | if (address(tokens[i].instance) == _tokenAddress) { 52 | return tokens[i]; 53 | } 54 | } 55 | revert("Token not found."); 56 | } 57 | 58 | function addToken(address _tokenAddress) external onlyOwner { 59 | for (uint i = 0; i < tokens.length; i++) { 60 | if (address(tokens[i].instance) == _tokenAddress) { 61 | revert('A token with the provided address already exists.'); 62 | } 63 | } 64 | tokens.push(generateTokenData(_tokenAddress)); 65 | } 66 | 67 | function deleteToken(address _tokenAddress) external onlyOwner { 68 | for (uint i = 0; i < tokens.length; i++) { 69 | if (address(tokens[i].instance) == _tokenAddress) { 70 | uint256 lastIdx = tokens.length - 1; 71 | tokens[i] = tokens[lastIdx]; 72 | delete lastIdx; 73 | break; 74 | } 75 | } 76 | revert('Token targeted for deletion not found.'); 77 | } 78 | 79 | function bet(address payable _referrer) payable external { 80 | if(!hasReferrer(msg.sender)) { 81 | addReferrer(_referrer); 82 | } 83 | bet(); 84 | } 85 | 86 | // value transfer tx based bet. 87 | function bet() payable public { 88 | // msg.value is added to the balance to begin with so you need to double it 89 | require(msg.value * 2 <= address(this).balance, 'Balance too low!'); 90 | 91 | // prevent "revert txn unless I won" attacks, see 92 | // https://developers.thundercore.com/docs/random-number-generator/ 93 | require(msg.sender == tx.origin); 94 | 95 | uint256 winnings = 0; 96 | 97 | if(RNGLibrary.rand() % 2 == 0) { 98 | // 3% is deducted to cover the referral bonus 99 | winnings = msg.value * 197/100; 100 | address(msg.sender).transfer(winnings); 101 | } 102 | 103 | payReferral(msg.value); 104 | emit BetSettled(msg.sender, winnings); 105 | } 106 | 107 | // This function handles ERC677 token bets. 108 | // This function does _not_ pay out referral bonuses. 109 | function betTokens(IBurnableMintableERC677Token _token, uint256 _amount) internal { 110 | // The smart contract needs to be able to pay double the value sent. 111 | require(_amount * 2 <= _token.balanceOf(address(this)), 'Contract balance too low!'); 112 | 113 | uint256 winnings = 0; 114 | 115 | if(RNGLibrary.rand() % 2 == 0) { 116 | winnings = _amount * 2; 117 | 118 | // This transaction can fail due to not enough gas being sent 119 | // with the call, so we specify the amount of gas to forward. 120 | _token.transfer.gas(50000)(tx.origin, winnings); 121 | } 122 | 123 | emit BetSettled(tx.origin, winnings); 124 | } 125 | 126 | // With no data sent, the contract token balance is simply updated. 127 | // Any data provided indicates that a user wants to make a bet. 128 | function onTokenTransfer(address /*_from*/, uint _value, bytes calldata _data) external returns(bool) { 129 | TokenData storage token = getToken(msg.sender); 130 | uint256 updatedBalance = token.instance.balanceOf(address(this)); 131 | 132 | if (_data.length == 0) { 133 | token.lastBalance = updatedBalance; 134 | } else { 135 | // It doesn't matter if the current stored `lastTokenBalance` is correct. 136 | // The balance increase needs to be _at least_ as much as the 137 | // stated _value. 138 | require(updatedBalance - token.lastBalance >= _value, "onTokenTransfer was called but no balance increase was detected."); 139 | token.lastBalance = updatedBalance; 140 | betTokens(token.instance, _value); 141 | } 142 | return true; 143 | } 144 | 145 | function withdraw(uint256 _amount) external onlyOwner { 146 | require(_amount <= address(this).balance, 'Balance too low!'); 147 | address payable owner = address(uint160(owner())); 148 | owner.transfer(_amount); 149 | } 150 | 151 | function withdrawTokens(address _tokenAddress, uint256 _amount) external onlyOwner { 152 | TokenData storage token = getToken(_tokenAddress); 153 | uint256 currentTokenBalance = token.instance.balanceOf(address(this)); 154 | 155 | require(_amount <= currentTokenBalance, 'Token balance too low!'); 156 | 157 | token.instance.transfer(owner(), _amount); 158 | token.lastBalance = currentTokenBalance - _amount; 159 | } 160 | 161 | // This fallback function eats all funds and gas sent to it. 162 | // The owner can withdraw from the contract balance via `withdraw` above. 163 | function() external payable {} 164 | } 165 | -------------------------------------------------------------------------------- /smart-contracts/contracts/ERC677Interface.sol: -------------------------------------------------------------------------------- 1 | import "openzeppelin-solidity/contracts/token/ERC20/ERC20.sol"; 2 | 3 | contract ERC677 is ERC20 { 4 | event Transfer(address indexed from, address indexed to, uint value, bytes data); 5 | function transferAndCall(address, uint, bytes calldata) external returns (bool); 6 | } 7 | 8 | contract IBurnableMintableERC677Token is ERC677 { 9 | function mint(address, uint256) public returns (bool); 10 | function burn(uint256 _value) public; 11 | function claimTokens(address _token, address _to) public; 12 | } 13 | 14 | contract ERC677Receiver { 15 | function onTokenTransfer(address _from, uint _value, bytes calldata _data) external returns(bool); 16 | } 17 | -------------------------------------------------------------------------------- /smart-contracts/contracts/Migrations.sol: -------------------------------------------------------------------------------- 1 | pragma solidity >=0.4.21 <0.6.0; 2 | 3 | contract Migrations { 4 | address public owner; 5 | uint public last_completed_migration; 6 | 7 | constructor() public { 8 | owner = msg.sender; 9 | } 10 | 11 | modifier restricted() { 12 | if (msg.sender == owner) _; 13 | } 14 | 15 | function setCompleted(uint completed) public restricted { 16 | last_completed_migration = completed; 17 | } 18 | 19 | function upgrade(address new_address) public restricted { 20 | Migrations upgraded = Migrations(new_address); 21 | upgraded.setCompleted(last_completed_migration); 22 | } 23 | } 24 | -------------------------------------------------------------------------------- /smart-contracts/contracts/ThunderRandomLibraryInterface.sol: -------------------------------------------------------------------------------- 1 | contract ThunderRandomLibraryInterface { 2 | function rand() public returns (uint256); 3 | } 4 | -------------------------------------------------------------------------------- /smart-contracts/environment.yml: -------------------------------------------------------------------------------- 1 | name: double-or-nothing 2 | channels: 3 | - conda-forge 4 | dependencies: 5 | - nodejs=10.16.3 # hub-analytics-frontend 6 | - yarn=1.22.0 7 | -------------------------------------------------------------------------------- /smart-contracts/migrations/1_initial_migration.js: -------------------------------------------------------------------------------- 1 | var Migrations = artifacts.require('./Migrations.sol') 2 | var Double = artifacts.require('./DoubleOrNothing.sol') 3 | 4 | module.exports = function(deployer) { 5 | deployer.deploy(Migrations) 6 | deployer.deploy( 7 | Double, 8 | 1000, 9 | 30, 10 | 86400, 11 | true, 12 | [600, 200, 100], 13 | [1, 500, 5, 750, 25, 1000], 14 | // [ 15 | // "0x42532085e51e5618575005f626589ff57d280d68", // tt-usdt testnet token address 16 | // "0xd60db41a718a73da844a4c454c8bd6e07173d722" // tt-dai testnet token address 17 | // ] 18 | [ 19 | '0x4f3C8E20942461e2c3Bdd8311AC57B0c222f2b82', // tt-usdt mainnet token address 20 | '0x2b31e3b88847f03c1335E99A0d1274A2c72059DE' // tt-dai mainnet token address 21 | ], 22 | '0xfe20903772D6df6DE5a42E00D011f1893bB0B772' 23 | ) 24 | } 25 | -------------------------------------------------------------------------------- /smart-contracts/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "DoubleOrNothingContract", 3 | "version": "1.0.0", 4 | "description": "Smart Contract for double or nothing", 5 | "license": "MIT", 6 | "scripts": { 7 | "migrate": "truffle migrate" 8 | }, 9 | "dependencies": { 10 | "@thundercore/referral-solidity": "^1.0.1", 11 | "openzeppelin-solidity": "^2.3.0", 12 | "truffle": "^5.1.22", 13 | "@truffle/hdwallet-provider": "^1.0.34" 14 | }, 15 | "devDependencies": { 16 | "truffle-flattener": "^1.4.2", 17 | "truffle-plugin-save-per-network-deployment-record": "https://github.com/thundercore/truffle-plugin-save-per-network-deployment-record.git" 18 | }, 19 | "engines": { 20 | "node": ">=8.0.0 <11.0.0" 21 | } 22 | } 23 | -------------------------------------------------------------------------------- /smart-contracts/truffle-config.js: -------------------------------------------------------------------------------- 1 | /** 2 | * Use this file to configure your truffle project. It's seeded with some 3 | * common settings for different networks and features like migrations, 4 | * compilation and testing. Uncomment the ones you need or modify 5 | * them to suit your project as necessary. 6 | * 7 | * More information about configuration can be found at: 8 | * 9 | * truffleframework.com/docs/advanced/configuration 10 | * 11 | */ 12 | 13 | const HDWalletProvider = require("@truffle/hdwallet-provider"); 14 | const fs = require("fs"); 15 | 16 | const TESTNET_PROVIDER = "https://testnet-rpc.thundercore.com"; 17 | const MAINNET_PROVIDER = "https://mainnet-rpc-partner.thundercore.com"; 18 | 19 | let privateKeys = null; 20 | let mnemonic = null; 21 | try { 22 | privateKeys = fs 23 | .readFileSync(".private-keys", { encoding: "ascii" }) 24 | .split("\n") 25 | .filter(x => x.length > 0); 26 | } catch (err) { 27 | if (err.code !== "ENOENT") { 28 | throw err; 29 | } 30 | } 31 | 32 | if (!privateKeys) { 33 | try { 34 | mnemonic = fs.readFileSync(".mnemonic", { encoding: "ascii" }).trim(); 35 | } catch (err) { 36 | if (err.code !== "ENOENT") { 37 | throw err; 38 | } 39 | } 40 | } 41 | 42 | module.exports = { 43 | networks: { 44 | // For `truffle develop` 45 | development: { 46 | host: "127.0.0.1", // Localhost (default: none) 47 | port: 9545, // Standard Ethereum port (default: none) 48 | network_id: "*" // Any network (default: none) 49 | }, 50 | "thunder-testnet": { 51 | provider: () => { 52 | if (privateKeys === null && mnemonic === null) { 53 | throw new Error("Please create a .private-keys or .mnemonic file"); 54 | } 55 | return privateKeys 56 | ? new HDWalletProvider( 57 | privateKeys, 58 | TESTNET_PROVIDER, 59 | 0, // <- change address_index if you want to default to an address other than the first one 60 | privateKeys.length 61 | ) 62 | : new HDWalletProvider( 63 | mnemonic, 64 | TESTNET_PROVIDER, 65 | 0 // <- change address_index if you want to use an address other than the first one 66 | ); 67 | }, 68 | network_id: "18" 69 | }, 70 | "thunder-mainnet": { 71 | provider: () => { 72 | if (privateKeys === null && mnemonic === null) { 73 | throw new Error("Please create a .private-keys or .mnemonic file"); 74 | } 75 | 76 | return privateKeys 77 | ? new HDWalletProvider( 78 | privateKeys, 79 | MAINNET_PROVIDER, 80 | 0, // <- change address_index if you want to use non-first address 81 | privateKeys.length 82 | ) 83 | : new HDWalletProvider( 84 | mnemonic, 85 | MAINNET_PROVIDER, 86 | 0 // <- change address_index if you want to use non-first address 87 | ); 88 | }, 89 | network_id: "108" 90 | } 91 | }, 92 | 93 | // Set default mocha options here, use special reporters etc. 94 | mocha: { 95 | // timeout: 100000 96 | }, 97 | 98 | // Configure your compilers 99 | compilers: { 100 | solc: { 101 | version: "0.5.0", // Fetch exact version from solc-bin (default: truffle's version) 102 | settings: { 103 | // see the solidity docs for advice about optimization and evmversion 104 | optimizer: { 105 | enabled: true, 106 | runs: 200 107 | }, 108 | evmVersion: "byzantium" // ThunderCore supports the "byzantium" hardfork as of PaLa release 2 109 | } 110 | } 111 | } 112 | }; 113 | --------------------------------------------------------------------------------