pages.js 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335
  1. const chalk = require('chalk');
  2. const common = require('../lib/common');
  3. const inquirer = require('../lib/inquirer');
  4. const shellExec = require('shell-exec')
  5. const files = require('../lib/files')
  6. const fs = require('fs')
  7. const getFilesIn = require('get-files-in')
  8. var shell = require('shelljs');
  9. let logger = require('perfect-logger');
  10. const globalVariables = require('../lib/globalVars');
  11. logger.info(process.platform + " detected")
  12. if (process.platform === 'win32' || process.platform === 'win64') {
  13. adbRun = 'adb'
  14. } else {
  15. adbRun = './adb'
  16. }
  17. logger.initialize('RunTIme', {
  18. logLevelFile: 0, // Log level for file
  19. logLevelConsole: -1, // Log level for STDOUT/STDERR
  20. logDirectory: 'data/', // Log directory
  21. });
  22. module.exports = {
  23. removeCompatibleApps: async () => {
  24. let installedAppList
  25. common.header('Remove Installed Apps')
  26. logger.info('Remove Installed Apps')
  27. await shellExec(adbRun + ' shell pm list packages -3').then(async function (result) {
  28. logger.info('Packages recieved from watch')
  29. if (result.stderr.includes('error')) {
  30. logger.info(result.stderr)
  31. console.log(chalk.red('Device not authorised'))
  32. common.pause(3000)
  33. await shellExec(adbRun + ' kill-server').then(async function (result) {
  34. logger.info('Restarting ADB')
  35. logger.info(result.stdout)
  36. console.log('Please reconnect to watch')
  37. common.pause(3000)
  38. logger.info('Remove Installed Apps Failed')
  39. module.exports.mainMenu()
  40. })
  41. } else {
  42. if (process.platform === 'win32' || process.platform === 'win64') {
  43. installedAppList = result.stdout.split('\r\n'); // split string on comma space
  44. installedAppList.splice(-1, 1)
  45. } else {
  46. installedAppList = result.stdout.split('\n'); // split string on comma space
  47. installedAppList.splice(-1, 1)
  48. }
  49. const value = await inquirer.installedApps(installedAppList);
  50. for (let element of value.removeAppsList) {
  51. console.log('Removing ' + element)
  52. logger.info('Removing ' + element)
  53. const package = element.substring(8)
  54. await shellExec(adbRun + ' uninstall ' + package).then(async function (result) {
  55. console.log(element + ' - ' + result.stdout);
  56. logger.info(element + ' - ' + result.stdout);
  57. });
  58. }
  59. console.log(chalk.green('Removed Selected User Apps'))
  60. logger.info('Removed Selected User Apps')
  61. await common.pause(2000)
  62. module.exports.mainMenu()
  63. }
  64. })
  65. },
  66. compatibleApps: async () => {
  67. logger.info("Compatible Apps")
  68. common.header('Install Compatible Apps')
  69. const compatibleApps = await common.getCompatibleAppsList()
  70. const value = await inquirer.compatibleApps();
  71. await shell.rm('-rf', './data/apps/*.apk');
  72. for (let element of value.removeAppsList) {
  73. for (let element2 of compatibleApps) {
  74. if (element === element2.name) {
  75. newName = element.replace(/\s/g, '');
  76. await common.downloadFile(element2.url, './data/apps/' + newName + '.apk')
  77. }
  78. }
  79. }
  80. const apkList = await getFilesIn('./data/apps', matchFiletypes = ['apk'], checkSubDirectories = false)
  81. for (let element of apkList) {
  82. console.log('Installing ' + element)
  83. logger.info('Installing ' + element)
  84. await shellExec(adbRun + ' install -r ' + element).then(async function (result) {
  85. if (result.stderr != '') {
  86. logger.info('Error ' + result.stderr);
  87. console.log(chalk.redBright('Error - Device not authorised'));
  88. }
  89. console.log(element + ' - ' + result.stdout);
  90. logger.info(element + ' - ' + result.stdout);
  91. if (element === "data\\apps\\simpleweather_base.apk") {
  92. await common.downloadFile('http://kithub.cf/Karl/MiWatchKleaner-APKs/raw/master/Others/simpleweather_split_config.armeabi_v7a.apk', './data/apps/simpleweather_split_config.armeabi_v7a.apk')
  93. await common.downloadFile('http://kithub.cf/Karl/MiWatchKleaner-APKs/raw/master/Others/simpleweather_split_config.xhdpi.apk', './data/apps/simpleweather_split_config.xhdpi.apk')
  94. await shellExec(adbRun + ' install-multiple "data\\apps\\simpleweather_base.apk" "data\\apps\\simpleweather_split_config.armeabi_v7a.apk" "data\\apps\\simpleweather_split_config.xhdpi.apk"').then(function (result) {
  95. console.log(result)
  96. console.log('simpleWeather Activated On Watch');
  97. logger.info('simpleWeather Activated On Watch');
  98. })
  99. }
  100. if (element === "data\\apps\\MoreLocale.apk") {
  101. await shellExec(adbRun + ' shell pm grant jp.co.c_lis.ccl.morelocale android.permission.CHANGE_CONFIGURATION').then(function (result) {
  102. console.log('moreLocale Activated On Watch');
  103. logger.info('moreLocale Activated On Watch');
  104. })
  105. }
  106. if (element === "data\\apps\\com.alberto.locale.apk") {
  107. await shellExec(adbRun + ' shell pm grant com.alberto.locale android.permission.CHANGE_CONFIGURATION && ' + adbRun + ' shell am start -n com.alberto.locale/com.alberto.locale.MainActivity && ' + adbRun + ' shell pm grant com.alberto.locale android.permission.CHANGE_CONFIGURATION').then(function (result) {
  108. console.log(result)
  109. console.log('Alberto Locale Activated On Watch');
  110. logger.info('Alberto Locale Activated On Watch');
  111. });
  112. }
  113. });
  114. }
  115. console.log(chalk.green('Compatible Apps Installed'))
  116. logger.info('Compatible Apps Installed')
  117. await common.pause(2000)
  118. module.exports.mainMenu()
  119. },
  120. removeApps: async () => {
  121. logger.info("Remove Apps")
  122. common.header('Remove Apps')
  123. const value = await inquirer.removeAppsList();
  124. for (let element of value.removeAppsList) {
  125. // await shellExec(adbRun + ' shell pm disable-user --0 ' + element).then(function (result) {
  126. await shellExec(adbRun + ' shell pm uninstall -k --user 0 ' + element).then(function (result) {
  127. if (result.stderr != '') {
  128. logger.info('Error ' + result.stderr);
  129. console.log(chalk.redBright('Error - Device not authorised'));
  130. } else {
  131. logger.info('Removing ' + element + ' - ' + result.stdout);
  132. console.log('Removing ' + element + ' - ' + result.stdout);
  133. }
  134. });
  135. }
  136. console.log(chalk.green('Removal Complete'))
  137. await common.pause(2000)
  138. logger.info("Remove Complete")
  139. module.exports.mainMenu()
  140. },
  141. restoreApps: async () => {
  142. logger.info("Restore Apps")
  143. common.header('Restore Apps')
  144. const value = await inquirer.removeAppsList();
  145. for (let element of value.removeAppsList) {
  146. await shellExec(adbRun + ' shell cmd package install-existing ' + element).then(function (result) {
  147. if (result.stderr != '') {
  148. logger.info('Error ' + result.stderr);
  149. console.log(chalk.redBright('Error - Device not authorised'));
  150. } else {
  151. logger.info('Restoring ' + element + ' - ' + result.stdout);
  152. console.log('Restoring ' + element + ' - ' + result.stdout);
  153. }
  154. });
  155. }
  156. console.log(chalk.green('Restore Complete'))
  157. await common.pause(2000)
  158. logger.info("Restore Apps Complete")
  159. module.exports.mainMenu()
  160. },
  161. connectWatch: async () => {
  162. logger.info("Connect to watch")
  163. common.header('Connect to watch')
  164. const value = await inquirer.connectionType()
  165. if (value.connection === "usb") {
  166. await shellExec(adbRun + ' kill-server').then(async function (result) {
  167. logger.info('Restarting ADB')
  168. logger.info(result.stdout)
  169. })
  170. await shellExec(adbRun + ' devices').then(async function (result) {
  171. console.log(result.stdout)
  172. if (result.stdout.includes('device', 15)) {
  173. console.log(chalk.green('MiWatch Connected via USB'))
  174. await common.pause(3000)
  175. logger.info("MiWatch connected")
  176. globalVariables.localUSB = "X"
  177. module.exports.mainMenu()
  178. } else {
  179. console.log(chalk.red('MiWatch not found'))
  180. logger.info("MiWatch not found")
  181. await common.pause(2000)
  182. console.log(chalk.white('Try Again'))
  183. await common.pause(1000)
  184. module.exports.connectWatch()
  185. }
  186. })
  187. }
  188. if (value.connection === "wifi") {
  189. const value = await inquirer.connectWifi();
  190. globalVariables.miWatchIpaddress = value.connectWifi
  191. await shellExec(adbRun + ' kill-server').then(async function (result) {
  192. logger.info('Restarting ADB')
  193. logger.info(result.stdout)
  194. })
  195. await shellExec(adbRun + ' connect ' + globalVariables.miWatchIpaddress).then(async function (result) {
  196. logger.info("Connect Wifi Result " + result.stdout)
  197. if (result.stdout.includes('already connected') || result.stdout.includes('connected to ')) {
  198. console.log(chalk.green('MiWatch Connected'))
  199. globalVariables.localUSB = ""
  200. await common.pause(3000)
  201. logger.info("Connect Wifi Complete")
  202. module.exports.mainMenu()
  203. } else {
  204. if (result.stdout.includes('failed to authenticate')) {
  205. console.log(chalk.redBright('MiWatch not authenticated'))
  206. logger.info('MiWatch not authenticated')
  207. } else {
  208. console.log(chalk.red(result.stdout))
  209. logger.info(result.stdout)
  210. }
  211. await common.pause(2000)
  212. console.log(chalk.white('Try Again'))
  213. await common.pause(1000)
  214. module.exports.connectWifi()
  215. }
  216. }).catch()
  217. }
  218. },
  219. oneClick: async () => {
  220. logger.info("1-Click Karl0ss Klean")
  221. common.header('1-Click Karl0ss Klean')
  222. const removalPackagesList = files.loadPackageList()
  223. for (let element of removalPackagesList.apps) {
  224. await shellExec(adbRun + ' shell pm uninstall -k --user 0 ' + element).then(function (result) {
  225. if (result.stderr != '') {
  226. logger.info('Error ' + result.stderr);
  227. console.log(chalk.redBright('Error - Device not authorised'));
  228. } else {
  229. logger.info('Removing ' + element + ' - ' + result.stdout);
  230. console.log('Removing ' + element + ' - ' + result.stdout);
  231. }
  232. });
  233. }
  234. console.log(chalk.green('Removal Complete'))
  235. await common.pause(2000)
  236. logger.info("Remove Complete")
  237. logger.info("Compatible Apps")
  238. await shell.rm('-rf', './data/apps/*.apk');
  239. const compatibleApps = await common.getCompatibleAppsList()
  240. for (const element of compatibleApps) {
  241. if (element.Klean === "X") {
  242. try {
  243. logger.info('Downloading Latest ' + element.name + ' Complete')
  244. newName = element.name.replace(/\s/g, '');
  245. await common.downloadFile(element.url, './data/apps/' + newName + '.apk')
  246. logger.info('Downloading Latest ' + element.name + ' Complete')
  247. } catch (error) {
  248. logger.info('Downloading Latest ' + element.name + ' FAILED')
  249. }
  250. }
  251. }
  252. const apkList = await getFilesIn('./data/apps', matchFiletypes = ['apk'], checkSubDirectories = false)
  253. for (let element of apkList) {
  254. console.log('Installing ' + element)
  255. logger.info('Installing ' + element)
  256. await shellExec(adbRun + ' install -r ' + element).then(async function (result) {
  257. if (result.stderr != '') {
  258. logger.info('Error ' + result.stderr);
  259. console.log(chalk.redBright('Error - Device not authorised'));
  260. }
  261. console.log(element + ' - ' + result.stdout);
  262. logger.info(element + ' - ' + result.stdout);
  263. });
  264. }
  265. console.log(chalk.green('Compatible Apps Installed'))
  266. logger.info('Compatible Apps Installed')
  267. await common.pause(2000)
  268. module.exports.mainMenu()
  269. },
  270. restoreAnyApp: async () => {
  271. logger.info("Restore Any App")
  272. common.header('Restore Any App')
  273. const value = await inquirer.restoreAnyApp();
  274. await shellExec(adbRun + ' shell cmd package install-existing ' + value.restoreAnyApp).then(function (result) {
  275. if (result.stderr != '') {
  276. logger.info('Error ' + result.stderr);
  277. console.log(chalk.redBright('Error - Device not authorised'));
  278. } else {
  279. logger.info('Restoring ' + value.restoreAnyApp + ' - ' + result.stdout);
  280. console.log('Restoring ' + value.restoreAnyApp + ' - ' + result.stdout);
  281. }
  282. });
  283. console.log(chalk.green('Restore Complete'))
  284. await common.pause(2000)
  285. logger.info("App Restore Complete")
  286. module.exports.mainMenu()
  287. },
  288. mainMenu: async () => {
  289. common.header('Main Menu')
  290. const mainMenuSelection = await inquirer.mainMenu();
  291. switch (mainMenuSelection.mainMenu) {
  292. case 'connect to miwatch':
  293. module.exports.connectWatch()
  294. break;
  295. case '1-click karl0ss klean':
  296. module.exports.oneClick()
  297. break;
  298. case 'remove xiaomi apps':
  299. module.exports.removeApps()
  300. break;
  301. case 'restore xiaomi apps':
  302. module.exports.restoreApps()
  303. break;
  304. case 'install compatible apps':
  305. module.exports.compatibleApps()
  306. break;
  307. case 'remove installed apps':
  308. module.exports.removeCompatibleApps()
  309. break;
  310. case 'restore any app':
  311. module.exports.restoreAnyApp()
  312. break;
  313. case 'quit':
  314. break;
  315. default:
  316. // code block
  317. }
  318. }
  319. };