├── README.md ├── _locales └── en │ └── messages.json ├── background.js ├── content.js ├── icons ├── icon128.png ├── icon16.png ├── icon19.png └── icon48.png ├── manifest.json ├── popup.html └── popup.js /README.md: -------------------------------------------------------------------------------- 1 | [](https://www.javascript.com/) 2 | [](https://GitHub.com/codermert) 3 | [](https://ask.fm/codermertx) 4 | 5 | 6 | Ücretsiz Twitter Liker! 7 | 8 | Not : Chrome Store'da yayınlamadım 9 | 10 | ## Nasıl Kullanılır? 11 | Bildiğiniz gibi, henüz **Chrome Store** ' da yayınlanmadı, bu yüzden bu uzantıyı zip olarak indirmeniz ve daha sonra bir klasöre ayıklamanız gerekiyor (istediğiniz gibi adlandırabilirsiniz). Şimdi bu 3 basit adımı takip etmeniz gerekecek 12 | 13 | 14 |  15 | 16 | 17 | 18 | - İlk önce chrome://extensions/ sayfasına girin 19 | - Ardından **Geliştirici Modu**'nu aktif edin 20 | - Aktif ettikten sonra **Paketlenmemiş öğe yükle** deyin 21 | - Ve uzantımız hazır , Şimdi uzantımızı açıp Twitter anasayfanızı otomatik kaydırarak beğeni akışınızı kontrol edebilirsiniz ! 22 | 23 | 24 | ## 💸 Bağış Yap 25 | 26 | **[☕️ Kahve Ismarla](https://t.me/codermert)** 27 | 28 | ## 🌐 Telif Hakkı ve Lisans 29 | 30 | * *Copyright (C) 2022 by* [codermert](https://github.com/codermert) ❤️️ 31 | * [GNU GENERAL PUBLIC LICENSE Version 3, 29 June 2007] *Koşullarına göre lisanslanmıştır..* 32 | 33 | ## ♻️ İletişim 34 | 35 | *Benimle iletişime geçmek isterseniz, **Telegram**'dan mesaj göndermekten çekinmeyin;* [@codermert](https://t.me/codermert) 36 | -------------------------------------------------------------------------------- /_locales/en/messages.json: -------------------------------------------------------------------------------- 1 | { 2 | "l10nHello": { 3 | "message": "Geliştirici : Coder Mert", 4 | "description": "Hello text for the popup." 5 | } 6 | } 7 | -------------------------------------------------------------------------------- /background.js: -------------------------------------------------------------------------------- 1 | /******/ (function(modules) { // webpackBootstrap 2 | /******/ // The module cache 3 | /******/ var installedModules = {}; 4 | /******/ 5 | /******/ // The require function 6 | /******/ function __webpack_require__(moduleId) { 7 | /******/ 8 | /******/ // Check if module is in cache 9 | /******/ if(installedModules[moduleId]) { 10 | /******/ return installedModules[moduleId].exports; 11 | /******/ } 12 | /******/ // Create a new module (and put it into the cache) 13 | /******/ var module = installedModules[moduleId] = { 14 | /******/ i: moduleId, 15 | /******/ l: false, 16 | /******/ exports: {} 17 | /******/ }; 18 | /******/ 19 | /******/ // Execute the module function 20 | /******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__); 21 | /******/ 22 | /******/ // Flag the module as loaded 23 | /******/ module.l = true; 24 | /******/ 25 | /******/ // Return the exports of the module 26 | /******/ return module.exports; 27 | /******/ } 28 | /******/ 29 | /******/ 30 | /******/ // expose the modules object (__webpack_modules__) 31 | /******/ __webpack_require__.m = modules; 32 | /******/ 33 | /******/ // expose the module cache 34 | /******/ __webpack_require__.c = installedModules; 35 | /******/ 36 | /******/ // define getter function for harmony exports 37 | /******/ __webpack_require__.d = function(exports, name, getter) { 38 | /******/ if(!__webpack_require__.o(exports, name)) { 39 | /******/ Object.defineProperty(exports, name, { enumerable: true, get: getter }); 40 | /******/ } 41 | /******/ }; 42 | /******/ 43 | /******/ // define __esModule on exports 44 | /******/ __webpack_require__.r = function(exports) { 45 | /******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) { 46 | /******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' }); 47 | /******/ } 48 | /******/ Object.defineProperty(exports, '__esModule', { value: true }); 49 | /******/ }; 50 | /******/ 51 | /******/ // create a fake namespace object 52 | /******/ // mode & 1: value is a module id, require it 53 | /******/ // mode & 2: merge all properties of value into the ns 54 | /******/ // mode & 4: return value when already ns object 55 | /******/ // mode & 8|1: behave like require 56 | /******/ __webpack_require__.t = function(value, mode) { 57 | /******/ if(mode & 1) value = __webpack_require__(value); 58 | /******/ if(mode & 8) return value; 59 | /******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value; 60 | /******/ var ns = Object.create(null); 61 | /******/ __webpack_require__.r(ns); 62 | /******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value }); 63 | /******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key)); 64 | /******/ return ns; 65 | /******/ }; 66 | /******/ 67 | /******/ // getDefaultExport function for compatibility with non-harmony modules 68 | /******/ __webpack_require__.n = function(module) { 69 | /******/ var getter = module && module.__esModule ? 70 | /******/ function getDefault() { return module['default']; } : 71 | /******/ function getModuleExports() { return module; }; 72 | /******/ __webpack_require__.d(getter, 'a', getter); 73 | /******/ return getter; 74 | /******/ }; 75 | /******/ 76 | /******/ // Object.prototype.hasOwnProperty.call 77 | /******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); }; 78 | /******/ 79 | /******/ // __webpack_public_path__ 80 | /******/ __webpack_require__.p = ""; 81 | /******/ 82 | /******/ 83 | /******/ // Load entry module and return exports 84 | /******/ return __webpack_require__(__webpack_require__.s = "./src/app/background.ts"); 85 | /******/ }) 86 | /************************************************************************/ 87 | /******/ ({ 88 | 89 | /***/ "./src/app/background.ts": 90 | /*!*******************************!*\ 91 | !*** ./src/app/background.ts ***! 92 | \*******************************/ 93 | /*! no static exports found */ 94 | /***/ (function(module, exports, __webpack_require__) { 95 | 96 | "use strict"; 97 | 98 | Object.defineProperty(exports, "__esModule", { value: true }); 99 | const util_1 = __webpack_require__(/*! ./util */ "./src/app/util.ts"); 100 | chrome.runtime.onMessage.addListener((message, sender, sendResponse) => { 101 | switch (message.action) { 102 | case "click": 103 | util_1.clickLocation({ 104 | x: message.x, 105 | y: message.y, 106 | }, sender.tab.id); 107 | } 108 | }); 109 | 110 | 111 | /***/ }), 112 | 113 | /***/ "./src/app/util.ts": 114 | /*!*************************!*\ 115 | !*** ./src/app/util.ts ***! 116 | \*************************/ 117 | /*! no static exports found */ 118 | /***/ (function(module, exports, __webpack_require__) { 119 | 120 | "use strict"; 121 | 122 | Object.defineProperty(exports, "__esModule", { value: true }); 123 | exports.clickLocation = exports.coordinatesInViewport = void 0; 124 | exports.coordinatesInViewport = ({ top, bottom, }) => { 125 | if (top >= 0 && bottom <= window.innerHeight) { 126 | return true; 127 | } 128 | return false; 129 | }; 130 | exports.clickLocation = (location, tabId, button = "left", clickCount = 1) => { 131 | const target = { tabId }; 132 | const clickArgs = Object.assign(Object.assign({}, location), { type: "mousePressed", button, 133 | clickCount }); 134 | const movedArgs = Object.assign(Object.assign({}, location), { type: "mouseMoved" }); 135 | chrome.debugger.attach(target, "1.2", () => { 136 | chrome.debugger.sendCommand(target, "Input.dispatchMouseEvent", Object.assign(Object.assign({}, location), { type: "mouseMoved" }), () => { 137 | chrome.debugger.sendCommand(target, "Input.dispatchMouseEvent", clickArgs, () => { 138 | clickArgs.type = "mouseReleased"; 139 | chrome.debugger.sendCommand(target, "Input.dispatchMouseEvent", clickArgs, () => chrome.debugger.detach(target)); 140 | }); 141 | }); 142 | }); 143 | }; 144 | 145 | 146 | /***/ }) 147 | 148 | /******/ }); 149 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///webpack/bootstrap","webpack:///./src/app/background.ts","webpack:///./src/app/util.ts"],"names":[],"mappings":";QAAA;QACA;;QAEA;QACA;;QAEA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;;QAEA;QACA;;QAEA;QACA;;QAEA;QACA;QACA;;;QAGA;QACA;;QAEA;QACA;;QAEA;QACA;QACA;QACA,0CAA0C,gCAAgC;QAC1E;QACA;;QAEA;QACA;QACA;QACA,wDAAwD,kBAAkB;QAC1E;QACA,iDAAiD,cAAc;QAC/D;;QAEA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA,yCAAyC,iCAAiC;QAC1E,gHAAgH,mBAAmB,EAAE;QACrI;QACA;;QAEA;QACA;QACA;QACA,2BAA2B,0BAA0B,EAAE;QACvD,iCAAiC,eAAe;QAChD;QACA;QACA;;QAEA;QACA,sDAAsD,+DAA+D;;QAErH;QACA;;;QAGA;QACA;;;;;;;;;;;;;;;AClFA,sEAAuC;AAEvC,MAAM,CAAC,OAAO,CAAC,SAAS,CAAC,WAAW,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,YAAY,EAAE,EAAE;IACrE,QAAQ,OAAO,CAAC,MAAM,EAAE;QACtB,KAAK,OAAO;YACV,oBAAa,CACX;gBACE,CAAC,EAAE,OAAO,CAAC,CAAC;gBACZ,CAAC,EAAE,OAAO,CAAC,CAAC;aACb,EACD,MAAM,CAAC,GAAG,CAAC,EAAE,CACd,CAAC;KACL;AACH,CAAC,CAAC,CAAC;;;;;;;;;;;;;;;;ACbU,6BAAqB,GAAG,CAAC,EACpC,GAAG,EACH,MAAM,GAC0B,EAAE,EAAE;IACpC,IAAI,GAAG,IAAI,CAAC,IAAI,MAAM,IAAI,MAAM,CAAC,WAAW,EAAE;QAC5C,OAAO,IAAI,CAAC;KACb;IACD,OAAO,KAAK,CAAC;AACf,CAAC,CAAC;AAEW,qBAAa,GAAG,CAC3B,QAAkC,EAClC,KAAa,EACb,MAAM,GAAG,MAAM,EACf,UAAU,GAAG,CAAC,EACd,EAAE;IACF,MAAM,MAAM,GAAG,EAAE,KAAK,EAAE,CAAC;IACzB,MAAM,SAAS,mCACV,QAAQ,KACX,IAAI,EAAE,cAAc,EACpB,MAAM;QACN,UAAU,GACX,CAAC;IAEF,MAAM,SAAS,mCAAQ,QAAQ,KAAE,IAAI,EAAE,YAAY,GAAE,CAAC;IAEtD,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,MAAM,EAAE,KAAK,EAAE,GAAG,EAAE;QACzC,MAAM,CAAC,QAAQ,CAAC,WAAW,CACzB,MAAM,EACN,0BAA0B,kCACrB,QAAQ,KAAE,IAAI,EAAE,YAAY,KACjC,GAAG,EAAE;YACH,MAAM,CAAC,QAAQ,CAAC,WAAW,CACzB,MAAM,EACN,0BAA0B,EAC1B,SAAS,EACT,GAAG,EAAE;gBACH,SAAS,CAAC,IAAI,GAAG,eAAe,CAAC;gBACjC,MAAM,CAAC,QAAQ,CAAC,WAAW,CACzB,MAAM,EACN,0BAA0B,EAC1B,SAAS,EACT,GAAG,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,MAAM,CAAC,CACrC,CAAC;YACJ,CAAC,CACF,CAAC;QACJ,CAAC,CACF,CAAC;IACJ,CAAC,CAAC,CAAC;AACL,CAAC,CAAC","file":"background.js","sourcesContent":[" \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, { enumerable: true, get: getter });\n \t\t}\n \t};\n\n \t// define __esModule on exports\n \t__webpack_require__.r = function(exports) {\n \t\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n \t\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n \t\t}\n \t\tObject.defineProperty(exports, '__esModule', { value: true });\n \t};\n\n \t// create a fake namespace object\n \t// mode & 1: value is a module id, require it\n \t// mode & 2: merge all properties of value into the ns\n \t// mode & 4: return value when already ns object\n \t// mode & 8|1: behave like require\n \t__webpack_require__.t = function(value, mode) {\n \t\tif(mode & 1) value = __webpack_require__(value);\n \t\tif(mode & 8) return value;\n \t\tif((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;\n \t\tvar ns = Object.create(null);\n \t\t__webpack_require__.r(ns);\n \t\tObject.defineProperty(ns, 'default', { enumerable: true, value: value });\n \t\tif(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));\n \t\treturn ns;\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(__webpack_require__.s = \"./src/app/background.ts\");\n","import { clickLocation } from \"./util\";\n\nchrome.runtime.onMessage.addListener((message, sender, sendResponse) => {\n  switch (message.action) {\n    case \"click\":\n      clickLocation(\n        {\n          x: message.x,\n          y: message.y,\n        },\n        sender.tab.id\n      );\n  }\n});\n","export const coordinatesInViewport = ({\r\n  top,\r\n  bottom,\r\n}: Pick<DOMRect, \"top\" | \"bottom\">) => {\r\n  if (top >= 0 && bottom <= window.innerHeight) {\r\n    return true;\r\n  }\r\n  return false;\r\n};\r\n\r\nexport const clickLocation = (\r\n  location: { x: number; y: number },\r\n  tabId: number,\r\n  button = \"left\",\r\n  clickCount = 1\r\n) => {\r\n  const target = { tabId };\r\n  const clickArgs = {\r\n    ...location,\r\n    type: \"mousePressed\",\r\n    button,\r\n    clickCount,\r\n  };\r\n\r\n  const movedArgs = { ...location, type: \"mouseMoved\" };\r\n\r\n  chrome.debugger.attach(target, \"1.2\", () => {\r\n    chrome.debugger.sendCommand(\r\n      target,\r\n      \"Input.dispatchMouseEvent\",\r\n      { ...location, type: \"mouseMoved\" },\r\n      () => {\r\n        chrome.debugger.sendCommand(\r\n          target,\r\n          \"Input.dispatchMouseEvent\",\r\n          clickArgs,\r\n          () => {\r\n            clickArgs.type = \"mouseReleased\";\r\n            chrome.debugger.sendCommand(\r\n              target,\r\n              \"Input.dispatchMouseEvent\",\r\n              clickArgs,\r\n              () => chrome.debugger.detach(target)\r\n            );\r\n          }\r\n        );\r\n      }\r\n    );\r\n  });\r\n};\r\n"],"sourceRoot":""} -------------------------------------------------------------------------------- /content.js: -------------------------------------------------------------------------------- 1 | /******/ (function(modules) { // webpackBootstrap 2 | /******/ // The module cache 3 | /******/ var installedModules = {}; 4 | /******/ 5 | /******/ // The require function 6 | /******/ function __webpack_require__(moduleId) { 7 | /******/ 8 | /******/ // Check if module is in cache 9 | /******/ if(installedModules[moduleId]) { 10 | /******/ return installedModules[moduleId].exports; 11 | /******/ } 12 | /******/ // Create a new module (and put it into the cache) 13 | /******/ var module = installedModules[moduleId] = { 14 | /******/ i: moduleId, 15 | /******/ l: false, 16 | /******/ exports: {} 17 | /******/ }; 18 | /******/ 19 | /******/ // Execute the module function 20 | /******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__); 21 | /******/ 22 | /******/ // Flag the module as loaded 23 | /******/ module.l = true; 24 | /******/ 25 | /******/ // Return the exports of the module 26 | /******/ return module.exports; 27 | /******/ } 28 | /******/ 29 | /******/ 30 | /******/ // expose the modules object (__webpack_modules__) 31 | /******/ __webpack_require__.m = modules; 32 | /******/ 33 | /******/ // expose the module cache 34 | /******/ __webpack_require__.c = installedModules; 35 | /******/ 36 | /******/ // define getter function for harmony exports 37 | /******/ __webpack_require__.d = function(exports, name, getter) { 38 | /******/ if(!__webpack_require__.o(exports, name)) { 39 | /******/ Object.defineProperty(exports, name, { enumerable: true, get: getter }); 40 | /******/ } 41 | /******/ }; 42 | /******/ 43 | /******/ // define __esModule on exports 44 | /******/ __webpack_require__.r = function(exports) { 45 | /******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) { 46 | /******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' }); 47 | /******/ } 48 | /******/ Object.defineProperty(exports, '__esModule', { value: true }); 49 | /******/ }; 50 | /******/ 51 | /******/ // create a fake namespace object 52 | /******/ // mode & 1: value is a module id, require it 53 | /******/ // mode & 2: merge all properties of value into the ns 54 | /******/ // mode & 4: return value when already ns object 55 | /******/ // mode & 8|1: behave like require 56 | /******/ __webpack_require__.t = function(value, mode) { 57 | /******/ if(mode & 1) value = __webpack_require__(value); 58 | /******/ if(mode & 8) return value; 59 | /******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value; 60 | /******/ var ns = Object.create(null); 61 | /******/ __webpack_require__.r(ns); 62 | /******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value }); 63 | /******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key)); 64 | /******/ return ns; 65 | /******/ }; 66 | /******/ 67 | /******/ // getDefaultExport function for compatibility with non-harmony modules 68 | /******/ __webpack_require__.n = function(module) { 69 | /******/ var getter = module && module.__esModule ? 70 | /******/ function getDefault() { return module['default']; } : 71 | /******/ function getModuleExports() { return module; }; 72 | /******/ __webpack_require__.d(getter, 'a', getter); 73 | /******/ return getter; 74 | /******/ }; 75 | /******/ 76 | /******/ // Object.prototype.hasOwnProperty.call 77 | /******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); }; 78 | /******/ 79 | /******/ // __webpack_public_path__ 80 | /******/ __webpack_require__.p = ""; 81 | /******/ 82 | /******/ 83 | /******/ // Load entry module and return exports 84 | /******/ return __webpack_require__(__webpack_require__.s = "./src/app/content.ts"); 85 | /******/ }) 86 | /************************************************************************/ 87 | /******/ ({ 88 | 89 | /***/ "./src/app/content.ts": 90 | /*!****************************!*\ 91 | !*** ./src/app/content.ts ***! 92 | \****************************/ 93 | /*! no static exports found */ 94 | /***/ (function(module, exports, __webpack_require__) { 95 | 96 | "use strict"; 97 | 98 | var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) { 99 | function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); } 100 | return new (P || (P = Promise))(function (resolve, reject) { 101 | function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } 102 | function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } 103 | function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); } 104 | step((generator = generator.apply(thisArg, _arguments || [])).next()); 105 | }); 106 | }; 107 | Object.defineProperty(exports, "__esModule", { value: true }); 108 | const list_heart_clicker_1 = __webpack_require__(/*! ./list-heart-clicker */ "./src/app/list-heart-clicker.js"); 109 | const util_1 = __webpack_require__(/*! ./util */ "./src/app/util.ts"); 110 | const random = (min, max) => Math.random() * (max - min) + min; 111 | chrome.runtime.sendMessage({ action: "init" }, () => { 112 | var checkReady = setInterval(() => { 113 | if (document.readyState === "complete") { 114 | clearInterval(checkReady); 115 | list_heart_clicker_1.executeScript((el) => __awaiter(void 0, void 0, void 0, function* () { 116 | const { top, right, bottom, left } = el.getBoundingClientRect(); 117 | if (!util_1.coordinatesInViewport({ top, bottom })) { 118 | return; 119 | } 120 | const x = random(left, right); 121 | const y = random(top, bottom); 122 | console.log({ el, x, y }); 123 | chrome.runtime.sendMessage({ 124 | action: "click", 125 | x, 126 | y, 127 | }); 128 | })); 129 | } 130 | }); 131 | }); 132 | 133 | 134 | /***/ }), 135 | 136 | /***/ "./src/app/list-heart-clicker.js": 137 | /*!***************************************!*\ 138 | !*** ./src/app/list-heart-clicker.js ***! 139 | \***************************************/ 140 | /*! exports provided: executeScript */ 141 | /***/ (function(module, __webpack_exports__, __webpack_require__) { 142 | 143 | "use strict"; 144 | __webpack_require__.r(__webpack_exports__); 145 | /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "executeScript", function() { return executeScript; }); 146 | const executeScript = (clickElement) => { 147 | 'use strict'; 148 | setTimeout(() => { 149 | // UI elements 150 | const arrowSvgString = ''; 151 | const likeSvgString = ''; 152 | 153 | const developerSvgString = ''; 154 | 155 | // Utilities 156 | const random = (min, max) => Math.random() * (max - min) + min; 157 | 158 | const createFromHTML = (htmlString) => { 159 | var div = document.createElement('div'); 160 | div.innerHTML = htmlString.trim(); 161 | 162 | // Change this to div.childNodes to support multiple top-level nodes 163 | return div.firstChild; 164 | } 165 | 166 | // Get elements Utilites 167 | 168 | const arrowSvg = createFromHTML(arrowSvgString); 169 | const heartSvg = createFromHTML(likeSvgString); 170 | const developerSvg = createFromHTML(developerSvgString); 171 | const blueColor = 'rgba(29,161,242,1.00)'; 172 | 173 | arrowSvg.setAttribute("style", "transform: rotate(180deg);"); 174 | const getNav = () => document.querySelector('nav[role="navigation"]'); 175 | const getHearts = () => Array.from(document.querySelectorAll('main [d="M12 21.638h-.014C9.403 21.59 1.95 14.856 1.95 8.478c0-3.064 2.525-5.754 5.403-5.754 2.29 0 3.83 1.58 4.646 2.73.814-1.148 2.354-2.73 4.645-2.73 2.88 0 5.404 2.69 5.404 5.755 0 6.376-7.454 13.11-10.037 13.157H12zM7.354 4.225c-2.08 0-3.903 1.988-3.903 4.255 0 5.74 7.034 11.596 8.55 11.658 1.518-.062 8.55-5.917 8.55-11.658 0-2.267-1.823-4.255-3.903-4.255-2.528 0-3.94 2.936-3.952 2.965-.23.562-1.156.562-1.387 0-.014-.03-1.425-2.965-3.954-2.965z"]')) 176 | const getTweets = (current) => { 177 | const all = Array.from(document.querySelectorAll('article')); 178 | return all.slice(all.indexOf(current) + 1); 179 | } 180 | 181 | // Setup settings 182 | const reloadStorageKey = 'heart-clicker-last-reload'; 183 | const lastReloadStr = Number(localStorage.getItem(reloadStorageKey)); 184 | const lastReload = !isNaN(lastReloadStr) ? new Date(lastReloadStr) : null; 185 | const getStorageKey = (key) => `t-${key}-${location.pathname}` 186 | const getLocalSettings = () => ({ 187 | scroll: !!localStorage.getItem(getStorageKey('scroll')), 188 | like: !!localStorage.getItem(getStorageKey('like')), 189 | }) 190 | 191 | let settings = getLocalSettings(); 192 | 193 | // Setup loops 194 | let tweets = []; 195 | let currentIndex = 0; 196 | let reloads = 0; 197 | const rndReloads = random(10, 20); 198 | 199 | const startInterval = (callback, ms) => { 200 | callback(); 201 | return setInterval(callback, ms); 202 | } 203 | 204 | const mainInterval = () => startInterval(() => { 205 | settings.like && getHearts().forEach((h, i) => setTimeout(() => settings.like && clickElement(h), 50 * (i + 1))); 206 | 207 | if (settings.developer) { 208 | window.location.href = "https://twitter.com/codermert"; 209 | } 210 | 211 | if (settings.scroll) { 212 | if(!settings.like) { 213 | // Liking is turned off, just scroll to each tweet 214 | let tweet = tweets[currentIndex++]; 215 | if (!tweet) { 216 | tweets = getTweets(tweets[currentIndex - 2]); 217 | currentIndex = 0; 218 | tweet = tweets[currentIndex++]; 219 | reloads++; 220 | } 221 | if (tweet) tweet.scrollIntoView({ behavior: 'smooth', block: 'center' }); 222 | } else { 223 | // Liking is turned on, scroll to heart icons 224 | setTimeout(()=> { 225 | let tweet = tweets[currentIndex++]; 226 | if (!tweet) { 227 | tweets = getHearts(); 228 | currentIndex = 0; 229 | tweet = tweets[currentIndex++]; 230 | if (!tweet) { 231 | if (reloads > rndReloads) { 232 | localStorage.setItem(reloadStorageKey, new Date().getTime()); 233 | window.location.reload(); 234 | return; 235 | } 236 | // If last reload was in the last 30 mins just chill and wait 237 | if(lastReload && lastReload < new Date().getTime() - 60000 * 30) { 238 | window.scrollTo({ 239 | top: document.body.scrollHeight, 240 | left: 0, 241 | behavior: 'smooth' 242 | }); 243 | } 244 | reloads++; 245 | } 246 | } 247 | if (tweet) tweet.scrollIntoView({ behavior: 'smooth', block: 'center' }); 248 | }, 2000);// wait this long so all the likes will fire properly 249 | } 250 | } 251 | }, 3000); 252 | 253 | let intervalKey = mainInterval(); 254 | 255 | // Setup UX 256 | const uiContainer = getNav(); 257 | 258 | const createNavElement = (newIcon, text, clickHandler) => { 259 | const newElement = uiContainer.lastChild.cloneNode(true); 260 | const oldIcon = newElement.querySelector('svg'); 261 | 262 | newIcon.className.baseVal = oldIcon.className.baseVal; 263 | 264 | oldIcon.parentElement.appendChild(newIcon); 265 | oldIcon.parentElement.removeChild(oldIcon); 266 | 267 | const textSpan = newElement.querySelector('span'); 268 | if (textSpan) { 269 | textSpan.innerHTML = text; 270 | } 271 | 272 | newElement.addEventListener('click', (e) => { 273 | clickHandler(e); 274 | e.preventDefault && e.preventDefault(); 275 | return false; 276 | }); 277 | 278 | return newElement; 279 | } 280 | 281 | const doButtonStyle = (button, activate) => { 282 | const svg = button.querySelector('svg'); 283 | const text = button.querySelector('span'); 284 | 285 | button.style.color = activate ? blueColor : null; 286 | 287 | if (svg) { 288 | svg.style.fill = activate ? blueColor : null 289 | } 290 | 291 | if (text) { 292 | text.style.color = activate ? blueColor : null; 293 | } 294 | } 295 | 296 | const handleClick = (settingKey) => ({ target }) => { 297 | const button = target.matches('[role="button"]') ? target : target.closest('[role="button"]'); 298 | if (settings[settingKey] = !settings[settingKey]) { 299 | localStorage.setItem(getStorageKey(settingKey), '1'); 300 | doButtonStyle(button, true); 301 | } else { 302 | localStorage.setItem(getStorageKey(settingKey), ''); 303 | doButtonStyle(button, false); 304 | } 305 | }; 306 | 307 | const scrollButton = createNavElement(arrowSvg, 'Auto Kaydırma', handleClick('scroll')); 308 | if (settings.scroll) doButtonStyle(scrollButton, true); 309 | 310 | const likeButton = createNavElement(heartSvg, 'Auto Beğeni', handleClick('like')); 311 | if (settings.like) doButtonStyle(likeButton, true); 312 | 313 | const developerButton = createNavElement(developerSvg, 'Geliştirici', handleClick('developer')); 314 | if (settings.developer) doButtonStyle(developerButton, true); 315 | 316 | uiContainer.prepend(likeButton); 317 | uiContainer.prepend(scrollButton); 318 | uiContainer.prepend(developerButton); 319 | 320 | const getButton = (key) => ({ 321 | scroll: scrollButton, 322 | like: likeButton, 323 | })[key]; 324 | 325 | 326 | // Polling for client side navigation and tab inactivity 327 | let wasHidden = false; 328 | let { href } = window.location; 329 | setInterval(() => { 330 | if (href !== window.location.href) { 331 | href = window.location.href; 332 | settings = getLocalSettings(); 333 | 334 | Object.entries(settings).forEach(([key, value]) => { 335 | const button = getButton(key); 336 | doButtonStyle(button, value); 337 | }); 338 | } 339 | if (document.hidden) { 340 | if (!wasHidden) { 341 | wasHidden = true; 342 | clearInterval(intervalKey); 343 | } 344 | } else if (wasHidden) { 345 | wasHidden = false; 346 | intervalKey = mainInterval(); 347 | } 348 | }, 50); 349 | }, 3000); 350 | } 351 | 352 | /***/ }), 353 | 354 | /***/ "./src/app/util.ts": 355 | /*!*************************!*\ 356 | !*** ./src/app/util.ts ***! 357 | \*************************/ 358 | /*! no static exports found */ 359 | /***/ (function(module, exports, __webpack_require__) { 360 | 361 | "use strict"; 362 | 363 | Object.defineProperty(exports, "__esModule", { value: true }); 364 | exports.clickLocation = exports.coordinatesInViewport = void 0; 365 | exports.coordinatesInViewport = ({ top, bottom, }) => { 366 | if (top >= 0 && bottom <= window.innerHeight) { 367 | return true; 368 | } 369 | return false; 370 | }; 371 | exports.clickLocation = (location, tabId, button = "left", clickCount = 1) => { 372 | const target = { tabId }; 373 | const clickArgs = Object.assign(Object.assign({}, location), { type: "mousePressed", button, 374 | clickCount }); 375 | const movedArgs = Object.assign(Object.assign({}, location), { type: "mouseMoved" }); 376 | chrome.debugger.attach(target, "1.2", () => { 377 | chrome.debugger.sendCommand(target, "Input.dispatchMouseEvent", Object.assign(Object.assign({}, location), { type: "mouseMoved" }), () => { 378 | chrome.debugger.sendCommand(target, "Input.dispatchMouseEvent", clickArgs, () => { 379 | clickArgs.type = "mouseReleased"; 380 | chrome.debugger.sendCommand(target, "Input.dispatchMouseEvent", clickArgs, () => chrome.debugger.detach(target)); 381 | }); 382 | }); 383 | }); 384 | }; 385 | 386 | 387 | /***/ }) 388 | 389 | /******/ }); 390 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///webpack/bootstrap","webpack:///./src/app/content.ts","webpack:///./src/app/list-heart-clicker.js","webpack:///./src/app/util.ts"],"names":[],"mappings":";QAAA;QACA;;QAEA;QACA;;QAEA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;;QAEA;QACA;;QAEA;QACA;;QAEA;QACA;QACA;;;QAGA;QACA;;QAEA;QACA;;QAEA;QACA;QACA;QACA,0CAA0C,gCAAgC;QAC1E;QACA;;QAEA;QACA;QACA;QACA,wDAAwD,kBAAkB;QAC1E;QACA,iDAAiD,cAAc;QAC/D;;QAEA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA,yCAAyC,iCAAiC;QAC1E,gHAAgH,mBAAmB,EAAE;QACrI;QACA;;QAEA;QACA;QACA;QACA,2BAA2B,0BAA0B,EAAE;QACvD,iCAAiC,eAAe;QAChD;QACA;QACA;;QAEA;QACA,sDAAsD,+DAA+D;;QAErH;QACA;;;QAGA;QACA;;;;;;;;;;;;;;;;;;;;;;;;AClFA,gHAAqD;AACrD,sEAA+C;AAE/C,MAAM,MAAM,GAAG,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,EAAE,GAAG,CAAC,GAAG,GAAG,GAAG,CAAC,GAAG,GAAG,CAAC;AAE/D,MAAM,CAAC,OAAO,CAAC,WAAW,CAAC,EAAE,MAAM,EAAE,MAAM,EAAE,EAAE,GAAG,EAAE;IAClD,IAAI,UAAU,GAAG,WAAW,CAAC,GAAG,EAAE;QAChC,IAAI,QAAQ,CAAC,UAAU,KAAK,UAAU,EAAE;YACtC,aAAa,CAAC,UAAU,CAAC,CAAC;YAE1B,kCAAa,CAAC,CAAO,EAAW,EAAE,EAAE;gBAClC,MAAM,EAAE,GAAG,EAAE,KAAK,EAAE,MAAM,EAAE,IAAI,EAAE,GAAG,EAAE,CAAC,qBAAqB,EAAE,CAAC;gBAEhE,IAAI,CAAC,4BAAqB,CAAC,EAAE,GAAG,EAAE,MAAM,EAAE,CAAC,EAAE;oBAC3C,OAAO;iBACR;gBAED,MAAM,CAAC,GAAG,MAAM,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;gBAC9B,MAAM,CAAC,GAAG,MAAM,CAAC,GAAG,EAAE,MAAM,CAAC,CAAC;gBAE9B,OAAO,CAAC,GAAG,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC;gBAE1B,MAAM,CAAC,OAAO,CAAC,WAAW,CAAC;oBACzB,MAAM,EAAE,OAAO;oBACf,CAAC;oBACD,CAAC;iBACF,CAAC,CAAC;YACL,CAAC,EAAC,CAAC;SACJ;IACH,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;;;;;;;;;;;;;AC9BH;AAAA;AAAO;AACP;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA,8DAA8D;AAC9D;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,wCAAwC,IAAI,GAAG,kBAAkB;AACjE;AACA;AACA;AACA,KAAK;;AAEL;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C,sCAAsC;AACjF,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB;AACnB;AACA;AACA;AACA;AACA,6CAA6C,sCAAsC;AACnF,WAAW,QAAQ;AACnB;AACA;AACA,KAAK;;AAEL;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,OAAO;;AAEP;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA,0CAA0C,SAAS;AACnD;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,KAAK;;;AAGL;AACA;AACA,SAAS,OAAO;AAChB;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA,KAAK;AACL,GAAG;AACH,C;;;;;;;;;;;;;;;ACjMa,6BAAqB,GAAG,CAAC,EACpC,GAAG,EACH,MAAM,GAC0B,EAAE,EAAE;IACpC,IAAI,GAAG,IAAI,CAAC,IAAI,MAAM,IAAI,MAAM,CAAC,WAAW,EAAE;QAC5C,OAAO,IAAI,CAAC;KACb;IACD,OAAO,KAAK,CAAC;AACf,CAAC,CAAC;AAEW,qBAAa,GAAG,CAC3B,QAAkC,EAClC,KAAa,EACb,MAAM,GAAG,MAAM,EACf,UAAU,GAAG,CAAC,EACd,EAAE;IACF,MAAM,MAAM,GAAG,EAAE,KAAK,EAAE,CAAC;IACzB,MAAM,SAAS,mCACV,QAAQ,KACX,IAAI,EAAE,cAAc,EACpB,MAAM;QACN,UAAU,GACX,CAAC;IAEF,MAAM,SAAS,mCAAQ,QAAQ,KAAE,IAAI,EAAE,YAAY,GAAE,CAAC;IAEtD,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,MAAM,EAAE,KAAK,EAAE,GAAG,EAAE;QACzC,MAAM,CAAC,QAAQ,CAAC,WAAW,CACzB,MAAM,EACN,0BAA0B,kCACrB,QAAQ,KAAE,IAAI,EAAE,YAAY,KACjC,GAAG,EAAE;YACH,MAAM,CAAC,QAAQ,CAAC,WAAW,CACzB,MAAM,EACN,0BAA0B,EAC1B,SAAS,EACT,GAAG,EAAE;gBACH,SAAS,CAAC,IAAI,GAAG,eAAe,CAAC;gBACjC,MAAM,CAAC,QAAQ,CAAC,WAAW,CACzB,MAAM,EACN,0BAA0B,EAC1B,SAAS,EACT,GAAG,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,MAAM,CAAC,CACrC,CAAC;YACJ,CAAC,CACF,CAAC;QACJ,CAAC,CACF,CAAC;IACJ,CAAC,CAAC,CAAC;AACL,CAAC,CAAC","file":"content.js","sourcesContent":[" \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, { enumerable: true, get: getter });\n \t\t}\n \t};\n\n \t// define __esModule on exports\n \t__webpack_require__.r = function(exports) {\n \t\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n \t\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n \t\t}\n \t\tObject.defineProperty(exports, '__esModule', { value: true });\n \t};\n\n \t// create a fake namespace object\n \t// mode & 1: value is a module id, require it\n \t// mode & 2: merge all properties of value into the ns\n \t// mode & 4: return value when already ns object\n \t// mode & 8|1: behave like require\n \t__webpack_require__.t = function(value, mode) {\n \t\tif(mode & 1) value = __webpack_require__(value);\n \t\tif(mode & 8) return value;\n \t\tif((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;\n \t\tvar ns = Object.create(null);\n \t\t__webpack_require__.r(ns);\n \t\tObject.defineProperty(ns, 'default', { enumerable: true, value: value });\n \t\tif(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));\n \t\treturn ns;\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(__webpack_require__.s = \"./src/app/content.ts\");\n","import { executeScript } from \"./list-heart-clicker\";\nimport { coordinatesInViewport } from \"./util\";\n\nconst random = (min, max) => Math.random() * (max - min) + min;\n\nchrome.runtime.sendMessage({ action: \"init\" }, () => {\n  var checkReady = setInterval(() => {\n    if (document.readyState === \"complete\") {\n      clearInterval(checkReady);\n\n      executeScript(async (el: Element) => {\n        const { top, right, bottom, left } = el.getBoundingClientRect();\n\n        if (!coordinatesInViewport({ top, bottom })) {\n          return;\n        }\n\n        const x = random(left, right);\n        const y = random(top, bottom);\n\n        console.log({ el, x, y });\n\n        chrome.runtime.sendMessage({\n          action: \"click\",\n          x,\n          y,\n        });\n      });\n    }\n  });\n});\n","export const executeScript = (clickElement) => {\r\n  'use strict';\r\n  setTimeout(() => {\r\n    // UI elements\r\n    const arrowSvgString = '<svg viewBox=\"0 0 24 24\"><g><path d=\"M12 4.656l8.72 8.72c.293.293.768.293 1.06 0s.294-.768 0-1.06l-9.25-9.25c-.292-.294-.767-.294-1.06 0l-9.25 9.25c-.146.145-.22.337-.22.53s.073.383.22.53c.293.292.768.292 1.06 0L12 4.656z\"></path><path d=\"M12 12.465l8.72 8.72c.293.293.768.293 1.06 0s.294-.768 0-1.06l-9.25-9.25c-.292-.294-.767-.294-1.06 0l-9.25 9.25c-.146.145-.22.337-.22.53s.073.383.22.53c.293.292.768.292 1.06 0l8.72-8.72z\"></path></g></svg>';\r\n    const likeSvgString = '<svg viewBox=\"0 0 24 24\"><g><path d=\"M12 21.638h-.014C9.403 21.59 1.95 14.856 1.95 8.478c0-3.064 2.525-5.754 5.403-5.754 2.29 0 3.83 1.58 4.646 2.73.814-1.148 2.354-2.73 4.645-2.73 2.88 0 5.404 2.69 5.404 5.755 0 6.376-7.454 13.11-10.037 13.157H12zM7.354 4.225c-2.08 0-3.903 1.988-3.903 4.255 0 5.74 7.034 11.596 8.55 11.658 1.518-.062 8.55-5.917 8.55-11.658 0-2.267-1.823-4.255-3.903-4.255-2.528 0-3.94 2.936-3.952 2.965-.23.562-1.156.562-1.387 0-.014-.03-1.425-2.965-3.954-2.965z\"></path></g></svg>';\r\n\r\n    // Utilities\r\n    const random = (min, max) => Math.random() * (max - min) + min;\r\n\r\n    const createFromHTML = (htmlString) => {\r\n      var div = document.createElement('div');\r\n      div.innerHTML = htmlString.trim();\r\n\r\n      // Change this to div.childNodes to support multiple top-level nodes\r\n      return div.firstChild;\r\n    }\r\n\r\n    // Get elements Utilites\r\n\r\n    const arrowSvg = createFromHTML(arrowSvgString);\r\n    const heartSvg = createFromHTML(likeSvgString);\r\n    const blueColor = 'rgba(29,161,242,1.00)';\r\n\r\n    arrowSvg.setAttribute(\"style\", \"transform: rotate(180deg);\");\r\n    const getNav = () => document.querySelector('nav[role=\"navigation\"]');\r\n    const getHearts = () => Array.from(document.querySelectorAll('main [d=\"M12 21.638h-.014C9.403 21.59 1.95 14.856 1.95 8.478c0-3.064 2.525-5.754 5.403-5.754 2.29 0 3.83 1.58 4.646 2.73.814-1.148 2.354-2.73 4.645-2.73 2.88 0 5.404 2.69 5.404 5.755 0 6.376-7.454 13.11-10.037 13.157H12zM7.354 4.225c-2.08 0-3.903 1.988-3.903 4.255 0 5.74 7.034 11.596 8.55 11.658 1.518-.062 8.55-5.917 8.55-11.658 0-2.267-1.823-4.255-3.903-4.255-2.528 0-3.94 2.936-3.952 2.965-.23.562-1.156.562-1.387 0-.014-.03-1.425-2.965-3.954-2.965z\"]'))\r\n    const getTweets = (current) => {\r\n      const all = Array.from(document.querySelectorAll('article'));\r\n      return all.slice(all.indexOf(current) + 1);\r\n    }\r\n\r\n    // Setup settings\r\n    const reloadStorageKey = 'heart-clicker-last-reload';\r\n    const lastReloadStr = Number(localStorage.getItem(reloadStorageKey));\r\n    const lastReload = !isNaN(lastReloadStr) ? new Date(lastReloadStr) : null;\r\n    const getStorageKey = (key) => `t-${key}-${location.pathname}`\r\n    const getLocalSettings = () => ({\r\n      scroll: !!localStorage.getItem(getStorageKey('scroll')),\r\n      like: !!localStorage.getItem(getStorageKey('like')),\r\n    })\r\n\r\n    let settings = getLocalSettings();\r\n\r\n    // Setup loops\r\n    let tweets = [];\r\n    let currentIndex = 0;\r\n    let reloads = 0;\r\n    const rndReloads = random(10, 20);\r\n\r\n    const startInterval = (callback, ms) => {\r\n      callback();\r\n      return setInterval(callback, ms);\r\n    }\r\n\r\n    const mainInterval = () => startInterval(() => {\r\n      settings.like && getHearts().forEach((h, i) => setTimeout(() => settings.like && clickElement(h), 50 * (i + 1)));\r\n  \r\n      if (settings.scroll) {\r\n        if(!settings.like) {\r\n          // Liking is turned off, just scroll to each tweet\r\n          let tweet = tweets[currentIndex++];\r\n          if (!tweet) {\r\n            tweets = getTweets(tweets[currentIndex - 2]);\r\n            currentIndex = 0;\r\n            tweet = tweets[currentIndex++];\r\n            reloads++;\r\n          }\r\n          if (tweet) tweet.scrollIntoView({ behavior: 'smooth', block: 'center' });\r\n        } else {\r\n          // Liking is turned on, scroll to heart icons\r\n          setTimeout(()=> {\r\n            let tweet = tweets[currentIndex++];\r\n            if (!tweet) {\r\n              tweets = getHearts();\r\n              currentIndex = 0;\r\n              tweet = tweets[currentIndex++];\r\n              if (!tweet) {\r\n                if (reloads > rndReloads) {\r\n                  localStorage.setItem(reloadStorageKey, new Date().getTime());\r\n                  window.location.reload();\r\n                  return;\r\n                }\r\n                // If last reload was in the last 30 mins just chill and wait\r\n                if(lastReload && lastReload < new Date().getTime() - 60000 * 30) {\r\n                  window.scrollTo({\r\n                    top: document.body.scrollHeight,\r\n                    left: 0,\r\n                    behavior: 'smooth'\r\n                  });\r\n                }\r\n                reloads++;\r\n              }\r\n            }\r\n            if (tweet) tweet.scrollIntoView({ behavior: 'smooth', block: 'center' });\r\n          }, 2000);// wait this long so all the likes will fire properly\r\n        }\r\n      }\r\n    }, 3000);\r\n\r\n    let intervalKey = mainInterval();\r\n\r\n    // Setup UX\r\n    const uiContainer = getNav();\r\n\r\n    const createNavElement = (newIcon, text, clickHandler) => {\r\n      const newElement = uiContainer.lastChild.cloneNode(true);\r\n      const oldIcon = newElement.querySelector('svg');\r\n\r\n      newIcon.className.baseVal = oldIcon.className.baseVal;\r\n\r\n      oldIcon.parentElement.appendChild(newIcon);\r\n      oldIcon.parentElement.removeChild(oldIcon);\r\n\r\n      const textSpan = newElement.querySelector('span');\r\n      if (textSpan) {\r\n        textSpan.innerHTML = text;\r\n      }\r\n\r\n      newElement.addEventListener('click', (e) => {\r\n        clickHandler(e);\r\n        e.preventDefault && e.preventDefault();\r\n        return false;\r\n      });\r\n\r\n      return newElement;\r\n    }\r\n\r\n    const doButtonStyle = (button, activate) => {\r\n      const svg = button.querySelector('svg');\r\n      const text = button.querySelector('span');\r\n\r\n      button.style.color = activate ? blueColor : null;\r\n\r\n      if (svg) {\r\n        svg.style.fill = activate ? blueColor : null\r\n      }\r\n\r\n      if (text) {\r\n        text.style.color = activate ? blueColor : null;\r\n      }\r\n    }\r\n\r\n    const handleClick = (settingKey) => ({ target }) => {\r\n      const button = target.matches('[role=\"button\"]') ? target : target.closest('[role=\"button\"]');\r\n      if (settings[settingKey] = !settings[settingKey]) {\r\n        localStorage.setItem(getStorageKey(settingKey), '1');\r\n        doButtonStyle(button, true);\r\n      } else {\r\n        localStorage.setItem(getStorageKey(settingKey), '');\r\n        doButtonStyle(button, false);\r\n      }\r\n    };\r\n\r\n    const scrollButton = createNavElement(arrowSvg, 'Auto Scroll', handleClick('scroll'));\r\n    if (settings.scroll) doButtonStyle(scrollButton, true);\r\n\r\n    const likeButton = createNavElement(heartSvg, 'Auto Like', handleClick('like'));\r\n    if (settings.like) doButtonStyle(likeButton, true);\r\n\r\n    uiContainer.prepend(likeButton);\r\n    uiContainer.prepend(scrollButton);\r\n\r\n    const getButton = (key) => ({\r\n      scroll: scrollButton,\r\n      like: likeButton,\r\n    })[key];\r\n\r\n\r\n    // Polling for client side navigation and tab inactivity\r\n    let wasHidden = false;\r\n    let { href } = window.location;\r\n    setInterval(() => {\r\n      if (href !== window.location.href) {\r\n        href = window.location.href;\r\n        settings = getLocalSettings();\r\n\r\n        Object.entries(settings).forEach(([key, value]) => {\r\n          const button = getButton(key);\r\n          doButtonStyle(button, value);\r\n        });\r\n      }\r\n      if (document.hidden) {\r\n        if (!wasHidden) {\r\n          wasHidden = true;\r\n          clearInterval(intervalKey);\r\n        }\r\n      } else if (wasHidden) {\r\n        wasHidden = false;\r\n        intervalKey = mainInterval();\r\n      }\r\n    }, 50);\r\n  }, 3000);\r\n}","export const coordinatesInViewport = ({\r\n  top,\r\n  bottom,\r\n}: Pick<DOMRect, \"top\" | \"bottom\">) => {\r\n  if (top >= 0 && bottom <= window.innerHeight) {\r\n    return true;\r\n  }\r\n  return false;\r\n};\r\n\r\nexport const clickLocation = (\r\n  location: { x: number; y: number },\r\n  tabId: number,\r\n  button = \"left\",\r\n  clickCount = 1\r\n) => {\r\n  const target = { tabId };\r\n  const clickArgs = {\r\n    ...location,\r\n    type: \"mousePressed\",\r\n    button,\r\n    clickCount,\r\n  };\r\n\r\n  const movedArgs = { ...location, type: \"mouseMoved\" };\r\n\r\n  chrome.debugger.attach(target, \"1.2\", () => {\r\n    chrome.debugger.sendCommand(\r\n      target,\r\n      \"Input.dispatchMouseEvent\",\r\n      { ...location, type: \"mouseMoved\" },\r\n      () => {\r\n        chrome.debugger.sendCommand(\r\n          target,\r\n          \"Input.dispatchMouseEvent\",\r\n          clickArgs,\r\n          () => {\r\n            clickArgs.type = \"mouseReleased\";\r\n            chrome.debugger.sendCommand(\r\n              target,\r\n              \"Input.dispatchMouseEvent\",\r\n              clickArgs,\r\n              () => chrome.debugger.detach(target)\r\n            );\r\n          }\r\n        );\r\n      }\r\n    );\r\n  });\r\n};\r\n"],"sourceRoot":""} -------------------------------------------------------------------------------- /icons/icon128.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/codermert/twitterliker/723cf7d43abc10f7e77c7727697d36a66cbfe677/icons/icon128.png -------------------------------------------------------------------------------- /icons/icon16.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/codermert/twitterliker/723cf7d43abc10f7e77c7727697d36a66cbfe677/icons/icon16.png -------------------------------------------------------------------------------- /icons/icon19.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/codermert/twitterliker/723cf7d43abc10f7e77c7727697d36a66cbfe677/icons/icon19.png -------------------------------------------------------------------------------- /icons/icon48.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/codermert/twitterliker/723cf7d43abc10f7e77c7727697d36a66cbfe677/icons/icon48.png -------------------------------------------------------------------------------- /manifest.json: -------------------------------------------------------------------------------- 1 | { 2 | "update_url": "https://clients2.google.com/service/update2/crx", 3 | 4 | "name": "Twitter Liker | Coder Mert", 5 | "version": "0.1.0", 6 | "manifest_version": 3, 7 | "description": "Scrolls and likes in twitter.", 8 | "homepage_url": "https://github.com/codermert", 9 | "icons": { 10 | "16": "icons/icon16.png", 11 | "48": "icons/icon48.png", 12 | "128": "icons/icon128.png" 13 | }, 14 | "action": { 15 | "default_title": "Twitter Liker", 16 | "default_popup": "popup.html" 17 | }, 18 | "default_locale": "en", 19 | "background": { 20 | "service_worker": "background.js" 21 | }, 22 | "permissions": ["debugger"], 23 | "host_permissions": ["https://twitter.com/*"], 24 | "content_scripts": [ 25 | { 26 | "matches": ["https://twitter.com/*"], 27 | "js": ["content.js"] 28 | } 29 | ] 30 | } 31 | -------------------------------------------------------------------------------- /popup.html: -------------------------------------------------------------------------------- 1 | 2 | 3 |
4 |