ServerItem.kt 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630
  1. package com.vpn.fastestvpnservice.customItems
  2. import android.content.res.Configuration
  3. import android.os.Looper
  4. import android.util.Log
  5. import android.widget.Toast
  6. import androidx.compose.foundation.BorderStroke
  7. import androidx.compose.foundation.Image
  8. import androidx.compose.foundation.background
  9. import androidx.compose.foundation.border
  10. import androidx.compose.foundation.clickable
  11. import androidx.compose.foundation.interaction.MutableInteractionSource
  12. import androidx.compose.foundation.layout.Arrangement
  13. import androidx.compose.foundation.layout.Box
  14. import androidx.compose.foundation.layout.Row
  15. import androidx.compose.foundation.layout.Spacer
  16. import androidx.compose.foundation.layout.fillMaxSize
  17. import androidx.compose.foundation.layout.fillMaxWidth
  18. import androidx.compose.foundation.layout.height
  19. import androidx.compose.foundation.layout.padding
  20. import androidx.compose.foundation.layout.size
  21. import androidx.compose.foundation.layout.wrapContentHeight
  22. import androidx.compose.foundation.layout.wrapContentWidth
  23. import androidx.compose.foundation.shape.CircleShape
  24. import androidx.compose.foundation.shape.RoundedCornerShape
  25. import androidx.compose.material3.Icon
  26. import androidx.compose.material3.IconButton
  27. import androidx.compose.material3.MaterialTheme
  28. import androidx.compose.material3.Surface
  29. import androidx.compose.material3.Text
  30. import androidx.compose.runtime.Composable
  31. import androidx.compose.runtime.getValue
  32. import androidx.compose.runtime.livedata.observeAsState
  33. import androidx.compose.runtime.mutableIntStateOf
  34. import androidx.compose.runtime.mutableStateOf
  35. import androidx.compose.runtime.remember
  36. import androidx.compose.runtime.rememberCoroutineScope
  37. import androidx.compose.runtime.saveable.rememberSaveable
  38. import androidx.compose.runtime.setValue
  39. import androidx.compose.ui.Alignment
  40. import androidx.compose.ui.Modifier
  41. import androidx.compose.ui.draw.alpha
  42. import androidx.compose.ui.draw.clip
  43. import androidx.compose.ui.draw.paint
  44. import androidx.compose.ui.draw.scale
  45. import androidx.compose.ui.graphics.Brush
  46. import androidx.compose.ui.graphics.Color
  47. import androidx.compose.ui.graphics.Shape
  48. import androidx.compose.ui.layout.ContentScale
  49. import androidx.compose.ui.platform.LocalContext
  50. import androidx.compose.ui.res.colorResource
  51. import androidx.compose.ui.res.painterResource
  52. import androidx.compose.ui.text.TextStyle
  53. import androidx.compose.ui.tooling.preview.Preview
  54. import androidx.compose.ui.unit.dp
  55. import androidx.compose.ui.unit.sp
  56. import androidx.lifecycle.viewmodel.compose.viewModel
  57. import androidx.navigation.NavHostController
  58. import androidx.navigation.compose.rememberNavController
  59. import com.stealthcopter.networktools.Ping
  60. import com.stealthcopter.networktools.ping.PingResult
  61. import com.stealthcopter.networktools.ping.PingStats
  62. import com.vpn.fastestvpnservice.R
  63. import com.vpn.fastestvpnservice.beans.Server
  64. import com.vpn.fastestvpnservice.beans.ServerList
  65. import com.vpn.fastestvpnservice.beans.favList
  66. import com.vpn.fastestvpnservice.beans.favListServer
  67. import com.vpn.fastestvpnservice.helpers.BasePreferenceHelper
  68. import com.vpn.fastestvpnservice.helpers.UIHelper
  69. import com.vpn.fastestvpnservice.screens.bottomNavBarScreens.onServer
  70. import com.vpn.fastestvpnservice.sealedClass.BottomBarScreen
  71. import com.vpn.fastestvpnservice.sealedClass.Screen
  72. import com.vpn.fastestvpnservice.utils.Utils
  73. import com.vpn.fastestvpnservice.viewmodels.HomeViewModel
  74. import com.vpn.fastestvpnservice.viewmodels.SearchListViewModel
  75. import com.vpn.fastestvpnservice.viewmodels.ServerListViewModel
  76. import com.vpn.fastestvpnservice.viewmodels.SplashViewModel
  77. import java.lang.Exception
  78. @Composable
  79. fun ServerItem(server: Server, navHostController: NavHostController, serverPing: Int = 0) {
  80. val context = LocalContext.current
  81. val scope = rememberCoroutineScope()
  82. var isServerClicked by remember { mutableStateOf(false) }
  83. val homeViewModel: HomeViewModel = viewModel{ HomeViewModel(context, scope) }
  84. val basePreferenceHelper = BasePreferenceHelper(context)
  85. var isFavorite by remember { mutableStateOf(server.isFavourited) }
  86. val isServerFavourited: Boolean = server.isFavourited == true
  87. Log.d("test-server_fav_d", "$isFavorite")
  88. Box(
  89. modifier = Modifier
  90. .fillMaxWidth()
  91. .background(color = MaterialTheme.colorScheme.background)
  92. .padding(bottom = 1.dp)
  93. ) {
  94. val serverListViewModel: ServerListViewModel = viewModel{
  95. ServerListViewModel(context)
  96. }
  97. val splashViewModel: SplashViewModel = viewModel{
  98. SplashViewModel(context)
  99. }
  100. if (isServerClicked) {
  101. Log.d("ServerCallbacks", "isServerClicked $isServerClicked")
  102. onServer.onServerSelected(
  103. context,
  104. homeViewModel,
  105. onClick = { isServerClicked = false },
  106. false,
  107. server
  108. )
  109. navHostController.popBackStack()
  110. }
  111. Row(
  112. verticalAlignment = Alignment.Top,
  113. horizontalArrangement = Arrangement.Start,
  114. modifier = Modifier
  115. .fillMaxWidth()
  116. .padding(start = 12.dp, end = 7.dp, top = 12.dp)
  117. .clickable(
  118. indication = null,
  119. interactionSource = remember { MutableInteractionSource() }
  120. ) {
  121. if (server.enable == 1) {
  122. basePreferenceHelper.setServerObject(server)
  123. isServerClicked = true
  124. } else {
  125. navHostController.navigate(
  126. Screen.Subscription.route
  127. )
  128. }
  129. }
  130. ) {
  131. var ping by remember { mutableIntStateOf(0) }
  132. // Ping.onAddress(server.ip as String).setTimeOutMillis(1000).doPing(
  133. // object : Ping.PingListener{
  134. // override fun onResult(pingResult: PingResult?) {
  135. // android.os.Handler(Looper.getMainLooper()).post {
  136. // ping = pingResult?.timeTaken?.toInt()!!
  137. // }
  138. // }
  139. //
  140. // override fun onError(e: Exception?) {}
  141. // override fun onFinished(pingStats: PingStats?) {}
  142. // }
  143. // )
  144. val icon = if (server.enable == 1) Utils.getDrawable(context, server.iso)
  145. else Utils.getDrawableGray(context, server.iso)
  146. Icon(
  147. painter = painterResource(id = icon),
  148. contentDescription = "Server Logo",
  149. tint = Color.Unspecified,
  150. modifier = Modifier
  151. .padding(bottom = 16.dp)
  152. .size(24.dp)
  153. .clip(CircleShape)
  154. .paint(
  155. painter = painterResource(id = icon),
  156. contentScale = ContentScale.FillHeight
  157. )
  158. )
  159. Text(text = server.server_name!!,
  160. style = MaterialTheme.typography.labelMedium,
  161. color = MaterialTheme.colorScheme.primary,
  162. modifier = Modifier
  163. .padding(start = 16.dp, bottom = 18.dp)
  164. .align(Alignment.CenterVertically)
  165. )
  166. Spacer(modifier = Modifier.weight(1F))
  167. Text(text = "$serverPing ms",
  168. style = MaterialTheme.typography.displayMedium,
  169. color = colorResource(id = R.color.blue_text),
  170. modifier = Modifier
  171. .padding(end = 30.dp, bottom = 18.dp)
  172. .align(Alignment.CenterVertically)
  173. )
  174. // Spacer(modifier = Modifier.weight(1F))
  175. IconButton(
  176. modifier = Modifier
  177. .padding(bottom = 18.dp, end = 8.dp)
  178. .size(25.dp),
  179. onClick = {
  180. isFavorite = !isFavorite!!
  181. serverListViewModel.favAndUnFav(server)
  182. }
  183. ) {
  184. Icon(
  185. painter = if (isServerFavourited) painterResource(
  186. id = R.drawable.fav_server3x) else painterResource(
  187. id = R.drawable.unfav_server3x),
  188. contentDescription = "Server Logo",
  189. tint = Color.Unspecified,
  190. modifier = Modifier.size(22.dp, 21.dp)
  191. )
  192. }
  193. }
  194. Surface(
  195. modifier = Modifier
  196. .padding(start = 0.dp, end = 0.dp)
  197. .height(1.dp)
  198. .fillMaxWidth()
  199. .alpha(0.6F)
  200. .align(Alignment.BottomCenter),
  201. color = colorResource(id = R.color.gray_icon)
  202. ) {}
  203. }
  204. }
  205. @Composable
  206. fun FavoriteServerItem(server: Server, navHostController: NavHostController) {
  207. val context = LocalContext.current
  208. val basePreferenceHelper = BasePreferenceHelper(context)
  209. var serverListViewModel: ServerListViewModel = viewModel {
  210. ServerListViewModel(context = context)
  211. }
  212. val splashViewModel: SplashViewModel = viewModel{
  213. SplashViewModel(context)
  214. }
  215. val scope = rememberCoroutineScope()
  216. val homeViewModel: HomeViewModel = viewModel {
  217. HomeViewModel(context, scope)
  218. }
  219. var isFavorite by remember { mutableStateOf(server.isFavourited) }
  220. val isServerFavourited: Boolean = server.isFavourited == true
  221. Log.d("test-server_fav_d", "$isFavorite")
  222. var isFavServerClicked by remember { mutableStateOf(false) }
  223. if (isFavServerClicked) {
  224. Log.d("ServerCallbacks", "isServerClicked $isFavServerClicked")
  225. onServer.onServerSelected(
  226. context,
  227. homeViewModel,
  228. onClick = { isFavServerClicked = false },
  229. true,
  230. server
  231. )
  232. navHostController.popBackStack(BottomBarScreen.Home.route, false)
  233. }
  234. Box(
  235. modifier = Modifier
  236. .fillMaxWidth()
  237. .background(MaterialTheme.colorScheme.background)
  238. .padding(bottom = 1.dp)
  239. ) {
  240. Row(
  241. verticalAlignment = Alignment.Top,
  242. horizontalArrangement = Arrangement.Start,
  243. modifier = Modifier
  244. .fillMaxWidth()
  245. .padding(start = 12.dp, end = 7.dp, top = 12.dp)
  246. .clickable(
  247. indication = null,
  248. interactionSource = remember { MutableInteractionSource() }
  249. ) {
  250. if (server.enable == 1) {
  251. isFavServerClicked = true
  252. } else {
  253. navHostController.navigate(
  254. Screen.Subscription.route
  255. )
  256. }
  257. }
  258. ) {
  259. var ping by remember { mutableIntStateOf(0) }
  260. // Ping.onAddress(server.ip as String).setTimeOutMillis(1000).doPing(
  261. // object : Ping.PingListener{
  262. // override fun onResult(pingResult: PingResult?) {
  263. // android.os.Handler(Looper.getMainLooper()).post {
  264. // ping = pingResult?.timeTaken?.toInt()!!
  265. // }
  266. // }
  267. //
  268. // override fun onError(e: Exception?) {}
  269. // override fun onFinished(pingStats: PingStats?) {}
  270. // }
  271. // )
  272. val icon = if (server.enable == 1) Utils.getDrawable(context, server.iso)
  273. else Utils.getDrawableGray(context, server.iso)
  274. Icon(
  275. painter = painterResource(id = icon),
  276. contentDescription = "Server Logo",
  277. tint = Color.Unspecified,
  278. modifier = Modifier
  279. .padding(bottom = 16.dp)
  280. .size(24.dp)
  281. .clip(CircleShape)
  282. .paint(
  283. painter = painterResource(id = icon),
  284. contentScale = ContentScale.FillHeight
  285. )
  286. )
  287. Text(text = server.server_name!!,
  288. color = MaterialTheme.colorScheme.primary,
  289. style = MaterialTheme.typography.labelMedium,
  290. modifier = Modifier
  291. .padding(start = 16.dp, bottom = 18.dp)
  292. .align(Alignment.CenterVertically)
  293. )
  294. Spacer(modifier = Modifier.weight(1F))
  295. Text(text = "${server.ping} ms",
  296. color = colorResource(id = R.color.blue_text),
  297. style = MaterialTheme.typography.displayMedium,
  298. modifier = Modifier
  299. .padding(end = 30.dp, bottom = 18.dp)
  300. .align(Alignment.CenterVertically)
  301. )
  302. // Spacer(modifier = Modifier.weight(1F))
  303. IconButton(
  304. onClick = {
  305. isFavorite = !isFavorite!!
  306. serverListViewModel.favAndUnFav(server)
  307. },
  308. modifier = Modifier
  309. .padding(bottom = 18.dp, end = 0.dp)
  310. .size(25.dp)
  311. // .clickable(
  312. // indication = null,
  313. // interactionSource = remember { MutableInteractionSource() }
  314. // ) { },
  315. ) {
  316. Icon(
  317. painter = if (isServerFavourited) painterResource(
  318. id = R.drawable.fav_server3x) else painterResource(
  319. id = R.drawable.unfav_server3x),
  320. contentDescription = "Server Logo",
  321. tint = Color.Unspecified,
  322. modifier = Modifier.size(22.dp, 21.dp)
  323. )
  324. }
  325. }
  326. Surface(
  327. modifier = Modifier
  328. .padding(start = 0.dp, end = 0.dp)
  329. .height(1.dp)
  330. .fillMaxWidth()
  331. .alpha(0.6F)
  332. .align(Alignment.BottomCenter)
  333. ,
  334. color = colorResource(id = R.color.gray_icon)
  335. ) {}
  336. }
  337. }
  338. @Composable
  339. fun ServerSearchItem(
  340. server: Server,
  341. navHostController: NavHostController,
  342. bgColor: Color = Color.White,
  343. textColor: Color = MaterialTheme.colorScheme.primary,
  344. isServerCallbackShown: Boolean = true,
  345. onServerDisable: () -> Unit
  346. ) {
  347. val context = LocalContext.current
  348. val scope = rememberCoroutineScope()
  349. val basePreferenceHelper = BasePreferenceHelper(context)
  350. val serverListViewModel: ServerListViewModel = viewModel {
  351. ServerListViewModel(context = context)
  352. }
  353. val splashViewModel: SplashViewModel = viewModel{
  354. SplashViewModel(context)
  355. }
  356. val searchListViewModel: SearchListViewModel = viewModel{
  357. SearchListViewModel(context, serverListViewModel, splashViewModel)
  358. }
  359. val homeViewModel: HomeViewModel = viewModel{
  360. HomeViewModel(context, scope)
  361. }
  362. var isSearchServerClicked by remember {
  363. mutableStateOf(false)
  364. }
  365. var isFavorite by remember { mutableStateOf(server.isFavourited) }
  366. val isServerFavourited: Boolean = server.isFavourited == true
  367. Log.d("test-server_fav_d", "$isFavorite")
  368. if (isSearchServerClicked) {
  369. Log.d("ServerCallbacks", "isServerClicked $isSearchServerClicked")
  370. onServer.onServerSelected(
  371. context,
  372. homeViewModel,
  373. onClick = { isSearchServerClicked = false },
  374. false,
  375. server
  376. )
  377. navHostController.popBackStack()
  378. }
  379. Box(
  380. modifier = Modifier
  381. .fillMaxWidth()
  382. .background(color = MaterialTheme.colorScheme.onBackground)
  383. .padding(bottom = 1.dp)
  384. ) {
  385. Row(
  386. verticalAlignment = Alignment.Top,
  387. horizontalArrangement = Arrangement.Start,
  388. modifier = Modifier
  389. .fillMaxWidth()
  390. .padding(start = 12.dp, end = 7.dp, top = 12.dp)
  391. .clickable(
  392. indication = null,
  393. interactionSource = remember { MutableInteractionSource() }
  394. ) {
  395. if (server.enable == 1) {
  396. basePreferenceHelper.setServerObject(server)
  397. isSearchServerClicked = isServerCallbackShown
  398. if (!isServerCallbackShown) {
  399. onServerDisable()
  400. }
  401. }
  402. else {
  403. navHostController.navigate(
  404. Screen.Subscription.route
  405. )
  406. }
  407. }
  408. ) {
  409. var ping by remember { mutableIntStateOf(0) }
  410. // Ping.onAddress(server.ip as String).setTimeOutMillis(1000).doPing(
  411. // object : Ping.PingListener{
  412. // override fun onResult(pingResult: PingResult?) {
  413. // android.os.Handler(Looper.getMainLooper()).post {
  414. // ping = pingResult?.timeTaken?.toInt()!!
  415. // }
  416. // }
  417. //
  418. // override fun onError(e: Exception?) {}
  419. // override fun onFinished(pingStats: PingStats?) {}
  420. // }
  421. // )
  422. val icon = if (server.enable == 1) Utils.getDrawable(context, server.iso)
  423. else Utils.getDrawableGray(context, server.iso)
  424. Icon(
  425. painter = painterResource(id = icon),
  426. contentDescription = "Server Logo",
  427. tint = Color.Unspecified,
  428. modifier = Modifier
  429. .padding(bottom = 16.dp)
  430. .size(24.dp)
  431. .clip(CircleShape)
  432. .paint(
  433. painter = painterResource(id = icon),
  434. contentScale = ContentScale.FillHeight
  435. )
  436. )
  437. Text(text = server.server_name!!,
  438. color = textColor,
  439. style = MaterialTheme.typography.labelMedium,
  440. modifier = Modifier
  441. .padding(start = 16.dp, bottom = 18.dp)
  442. .align(Alignment.CenterVertically)
  443. )
  444. Spacer(modifier = Modifier.weight(1F))
  445. Text(text = "${server.ping} ms",
  446. color = colorResource(id = R.color.blue_text),
  447. style = MaterialTheme.typography.displayMedium,
  448. modifier = Modifier
  449. .padding(end = 30.dp, bottom = 18.dp)
  450. .align(Alignment.CenterVertically)
  451. )
  452. IconButton(
  453. modifier = Modifier
  454. .padding(bottom = 18.dp, end = 8.dp)
  455. .size(25.dp),
  456. onClick = {
  457. isFavorite = !isFavorite!!
  458. serverListViewModel.favAndUnFav(server)
  459. }) {
  460. Icon(
  461. painter = if (isServerFavourited) painterResource(
  462. id = R.drawable.fav_server3x) else painterResource(
  463. id = R.drawable.unfav_server3x),
  464. contentDescription = "Server Logo",
  465. tint = Color.Unspecified,
  466. modifier = Modifier.size(22.dp, 21.dp),
  467. )
  468. }
  469. }
  470. Surface(
  471. modifier = Modifier
  472. .padding(start = 0.dp, end = 0.dp)
  473. .height(1.dp)
  474. .fillMaxWidth()
  475. .alpha(0.6F)
  476. .align(Alignment.BottomCenter),
  477. color = colorResource(id = R.color.gray_icon)
  478. ) {}
  479. }
  480. }
  481. @Composable
  482. fun ServerSpecificItem(
  483. server: Server,
  484. navHostController: NavHostController,
  485. onServerDisable: () -> Unit
  486. ) {
  487. val context = LocalContext.current
  488. val scope = rememberCoroutineScope()
  489. val basePreferenceHelper = BasePreferenceHelper(context)
  490. val serverListViewModel: ServerListViewModel = viewModel {
  491. ServerListViewModel(context = context)
  492. }
  493. val splashViewModel: SplashViewModel = viewModel{
  494. SplashViewModel(context)
  495. }
  496. val searchListViewModel: SearchListViewModel = viewModel{
  497. SearchListViewModel(context, serverListViewModel, splashViewModel)
  498. }
  499. val homeViewModel: HomeViewModel = viewModel{
  500. HomeViewModel(context, scope)
  501. }
  502. var isSearchServerClicked by remember {
  503. mutableStateOf(false)
  504. }
  505. Box(
  506. modifier = Modifier
  507. .fillMaxWidth()
  508. .background(color = MaterialTheme.colorScheme.background)
  509. .padding(bottom = 1.dp)
  510. ) {
  511. Row(
  512. verticalAlignment = Alignment.Top,
  513. horizontalArrangement = Arrangement.Start,
  514. modifier = Modifier
  515. .fillMaxWidth()
  516. .padding(start = 12.dp, end = 7.dp, top = 12.dp)
  517. .clickable(
  518. indication = null,
  519. interactionSource = remember { MutableInteractionSource() }
  520. ) {
  521. if (server.enable == 1) {
  522. basePreferenceHelper.setServerObject(server)
  523. onServerDisable()
  524. } else {
  525. navHostController.navigate(
  526. Screen.Subscription.route
  527. )
  528. }
  529. }
  530. ) {
  531. val icon = if (server.enable == 1) Utils.getDrawable(context, server.iso)
  532. else Utils.getDrawableGray(context, server.iso)
  533. Icon(
  534. painter = painterResource(id = icon),
  535. contentDescription = "Server Logo",
  536. tint = Color.Unspecified,
  537. modifier = Modifier
  538. .padding(bottom = 16.dp)
  539. .size(24.dp)
  540. .clip(CircleShape)
  541. .paint(
  542. painter = painterResource(id = icon),
  543. contentScale = ContentScale.FillHeight
  544. )
  545. )
  546. Text(text = server.server_name!!,
  547. color = MaterialTheme.colorScheme.primary,
  548. style = MaterialTheme.typography.labelMedium,
  549. modifier = Modifier
  550. .padding(start = 16.dp, bottom = 18.dp)
  551. .align(Alignment.CenterVertically)
  552. )
  553. }
  554. Surface(
  555. modifier = Modifier
  556. .padding(start = 0.dp, end = 0.dp)
  557. .height(1.dp)
  558. .fillMaxWidth()
  559. .alpha(0.6F)
  560. .align(Alignment.BottomCenter)
  561. ,
  562. color = colorResource(id = R.color.gray_icon)
  563. ) {}
  564. }
  565. }
  566. @Preview
  567. @Composable
  568. fun ServerItemPreview() {
  569. ServerItem(server = favListServer[0], rememberNavController())
  570. }
  571. @Preview
  572. @Composable
  573. fun FavoriteServerItemPreview() {
  574. FavoriteServerItem(server = favListServer[0], rememberNavController())
  575. }
  576. @Preview(uiMode = Configuration.UI_MODE_NIGHT_YES)
  577. @Composable
  578. fun FavoriteServerItemPreviewDark() {
  579. FavoriteServerItem(server = favListServer[0], rememberNavController())
  580. }
  581. @Preview
  582. @Composable
  583. fun ServerSearchItemPreview() {
  584. ServerSearchItem(server = favListServer[0], rememberNavController()) {}
  585. }