Initial Commit

main
Matthew 12 months ago
parent 4a77924fb4
commit 3fc7031177

1
app/.gitignore vendored

@ -0,0 +1 @@
/build

@ -0,0 +1,63 @@
plugins {
id 'com.android.application'
}
android {
compileSdk 32
defaultConfig {
applicationId "com.xypower.frpandroid"
minSdk 25
targetSdk 32
versionCode 1
versionName "1.0"
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
buildConfigField("String", "FrpVersion", '"0.56.0"')
buildConfigField("String", "FrpcFileName", '"libfrpc.so"')
buildConfigField("String", "LogFileName", '"frpc.log"')
buildConfigField("String", "ConfigFileName", '"config.toml"')
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
}
}
compileOptions {
sourceCompatibility JavaVersion.VERSION_1_8
targetCompatibility JavaVersion.VERSION_1_8
}
sourceSets {
main {
jniLibs.srcDirs = ['libs']
}
}
packagingOptions {
jniLibs {
useLegacyPackaging true
}
}
splits {
abi {
enable true
reset()
include "arm64-v8a", "x86_64", "armeabi-v7a"
universalApk true
}
}
}
dependencies {
implementation 'com.android.support:appcompat-v7:28.0.0'
implementation 'com.android.support.constraint:constraint-layout:2.0.4'
implementation 'org.jetbrains:annotations:15.0'
testImplementation 'junit:junit:4.13.2'
androidTestImplementation 'com.android.support.test:runner:1.0.2'
androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'
}

Binary file not shown.

Binary file not shown.

Binary file not shown.

@ -0,0 +1,21 @@
# Add project specific ProGuard rules here.
# You can control the set of applied configuration files using the
# proguardFiles setting in build.gradle.
#
# For more details, see
# http://developer.android.com/guide/developing/tools/proguard.html
# If your project uses WebView with JS, uncomment the following
# and specify the fully qualified class name to the JavaScript interface
# class:
#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
# public *;
#}
# Uncomment this to preserve the line number information for
# debugging stack traces.
#-keepattributes SourceFile,LineNumberTable
# If you keep the line number information, uncomment this to
# hide the original source file name.
#-renamesourcefileattribute SourceFile

@ -0,0 +1,43 @@
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
package="com.xypower.frpandroid">
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.FOREGROUND_SERVICE" />
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
<uses-permission android:name="android.permission.POST_NOTIFICATIONS" />
<uses-permission android:name="android.permission.FOREGROUND_SERVICE_SPECIAL_USE" />
<application
android:allowBackup="true"
android:dataExtractionRules="@xml/data_extraction_rules"
android:fullBackupContent="@xml/backup_rules"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/Theme.FrpAndroid"
tools:targetApi="31">
<activity
android:name=".ConfigActivity"
android:exported="false" />
<service
android:name=".ShellService"
android:enabled="true"
android:exported="true" />
<activity
android:name=".MainActivity"
android:exported="true">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>

@ -0,0 +1,20 @@
serverAddr = "61.169.135.146"
serverPort = 7000
dnsServer = "114.114.114.114"
auth.method = "token"
auth.token = "mint"
transport.tls.enable=false
transport.tls.disableCustomTLSFirstByte = false
[log]
level = "debug"
disablePrintColor = true
[[proxies]]
name = "p2p_adb"
type = "xtcp"
secretKey = "123456"
localIP = "127.0.0.1"
localPort = 5555

@ -0,0 +1,125 @@
package com.xypower.frpandroid;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import org.jetbrains.annotations.Nullable;
import java.io.BufferedReader;
import java.io.Closeable;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.Reader;
import java.nio.charset.Charset;
public class ConfigActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_config);
Button saveConfigButton = (Button)this.findViewById(R.id.saveConfigButton);
saveConfigButton.setOnClickListener((View.OnClickListener)(new View.OnClickListener() {
public final void onClick(View it) {
ConfigActivity.this.saveConfig();
ConfigActivity.this.finish();
}
}));
Button dontSaveConfigButton = (Button)this.findViewById(R.id.dontSaveConfigButton);
dontSaveConfigButton.setOnClickListener((View.OnClickListener)(new View.OnClickListener() {
public final void onClick(View it) {
ConfigActivity.this.finish();
}
}));
this.readConfig();
}
public final void readConfig() {
String[] files = this.fileList();
if (files == null) {
return;
}
EditText configEditText = (EditText)this.findViewById(R.id.configEditText);
configEditText.setText((CharSequence)"");
FileInputStream fileInputStream = null;
try {
fileInputStream = openFileInput("config.toml");
} catch (FileNotFoundException e) {
e.printStackTrace();
}
if (fileInputStream != null) {
InputStream var4 = (InputStream)fileInputStream;
// Charset var5 = Charsets.UTF_8;
Reader reader = (Reader)(new InputStreamReader(var4, Charset.defaultCharset()));
short var7 = 8192;
BufferedReader mReader = reader instanceof BufferedReader ? (BufferedReader)reader : new BufferedReader(reader, var7);
StringBuffer mRespBuff = new StringBuffer();
char[] buff = new char[1024];
boolean var13 = false;
while(true) {
int var14 = 0;
try {
var14 = mReader.read(buff);
} catch (IOException e) {
e.printStackTrace();
}
if (var14 == -1) {
try {
mReader.close();
} catch (IOException e) {
}
configEditText.setText((CharSequence)mRespBuff.toString());
break;
}
mRespBuff.append(buff, 0, var14);
}
}
}
private static final void closeFinally(Closeable closable) {
if (closable != null) {
try {
closable.close();
} catch (Throwable var3) {
// ExceptionsKt.addSuppressed(cause, var3);
}
}
}
public final void saveConfig() {
EditText configEditText = (EditText)this.findViewById(R.id.configEditText);
FileOutputStream fileOutputStream = null;
try {
fileOutputStream = this.openFileOutput("config.toml", 0);
} catch (FileNotFoundException e) {
e.printStackTrace();
}
try {
// Intrinsics.checkNotNullExpressionValue(configEditText, "configEditText");
String var6 = configEditText.getText().toString();
byte[] var8 = var6.getBytes(Charset.defaultCharset());
// Intrinsics.checkNotNullExpressionValue(var8, "this as java.lang.String).getBytes(charset)");
fileOutputStream.write(var8);
// Unit var13 = Unit.INSTANCE;
} catch (Throwable ex) {
ex.printStackTrace();
} finally {
closeFinally(fileOutputStream);
}
}
}

@ -0,0 +1,259 @@
package com.xypower.frpandroid;
import android.app.ActivityManager;
import android.app.NotificationChannel;
import android.app.NotificationManager;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
import android.content.SharedPreferences;
import android.content.pm.PackageManager;
import android.content.res.AssetManager;
import android.content.res.Resources;
import android.os.IBinder;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.support.v7.widget.SwitchCompat;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.CompoundButton;
import android.widget.Switch;
import android.widget.TextView;
import org.jetbrains.annotations.NotNull;
import java.io.Closeable;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.util.Arrays;
import java.util.Iterator;
public class MainActivity extends AppCompatActivity {
private Switch state_switch;
private Switch auto_start_switch;
private ShellService mService;
private boolean mBound;
private final ServiceConnection connection = new ServiceConnection() {
public void onServiceConnected(@NotNull ComponentName className, @NotNull IBinder service) {
// Intrinsics.checkNotNullParameter(className, "className");
// Intrinsics.checkNotNullParameter(service, "service");
ShellService.LocalBinder binder = (ShellService.LocalBinder)service;
MainActivity.this.mService = binder.getService();
MainActivity.this.mBound = true;
MainActivity.this.state_switch.setChecked(true);
}
public void onServiceDisconnected(@NotNull ComponentName arg0) {
// Intrinsics.checkNotNullParameter(arg0, "arg0");
MainActivity.this.mBound = false;
MainActivity.this.state_switch.setChecked(false);
}
};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
String versionName = null;
try {
versionName = this.getPackageManager().getPackageInfo(this.getPackageName(), 0).versionName;
} catch (PackageManager.NameNotFoundException e) {
e.printStackTrace();
}
TextView titleText = (TextView)this.findViewById(R.id.titleText);
// Intrinsics.checkNotNullExpressionValue(titleText, "titleText");
titleText.setText((CharSequence)("frp for Android - " + versionName + "/0.58.0"));
this.checkConfig();
this.checkNotificationPermission();
this.createBGNotificationChannel();
this.mBound = this.isServiceRunning(ShellService.class);
state_switch = (Switch)this.findViewById(R.id.state_switch);
// Intrinsics.checkNotNullExpressionValue(var10001, "findViewById<SwitchCompat>(R.id.state_switch)");
state_switch.setChecked(this.mBound);
state_switch.setOnCheckedChangeListener((CompoundButton.OnCheckedChangeListener)(new CompoundButton.OnCheckedChangeListener() {
public final void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
if (isChecked) {
MainActivity.this.startShell();
} else {
MainActivity.this.stopShell();
}
}
}));
final SharedPreferences editor = this.getSharedPreferences("data", 0);
auto_start_switch = (Switch) this.findViewById(R.id.auto_start_switch);
// Intrinsics.checkNotNullExpressionValue(var10001, "findViewById<SwitchCompat>(R.id.auto_start_switch)");
auto_start_switch.setChecked(editor.getBoolean("auto_start", false));
auto_start_switch.setOnCheckedChangeListener((CompoundButton.OnCheckedChangeListener)(new CompoundButton.OnCheckedChangeListener() {
public final void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
SharedPreferences.Editor editorx = editor.edit();
editorx.putBoolean("auto_start", isChecked);
editorx.apply();
}
}));
if (this.mBound) {
Intent intent = new Intent((Context)this, ShellService.class);
this.bindService(intent, (ServiceConnection)this.connection, 1);
}
this.setListener();
}
private final void setListener() {
Button configButton = (Button)this.findViewById(R.id.configButton);
configButton.setOnClickListener((View.OnClickListener)(new View.OnClickListener() {
public final void onClick(View it) {
Intent intent = new Intent((Context)MainActivity.this, ConfigActivity.class);
MainActivity.this.startActivity(intent);
}
}));
Button refreshButton = (Button)this.findViewById(R.id.refreshButton);
refreshButton.setOnClickListener((View.OnClickListener)(new View.OnClickListener() {
public final void onClick(View it) {
MainActivity.this.readLog();
}
}));
Button deleteButton = (Button)this.findViewById(R.id.deleteButton);
deleteButton.setOnClickListener((View.OnClickListener)(new View.OnClickListener() {
public final void onClick(View it) {
MainActivity.this.mService.clearOutput();
MainActivity.this.readLog();
}
}));
}
public final void readLog() {
TextView logTextView = (TextView)this.findViewById(R.id.logTextView);
if (this.mBound) {
// Intrinsics.checkNotNullExpressionValue(logTextView, "logTextView");
ShellService var10001 = this.mService;
if (var10001 == null) {
// Intrinsics.throwUninitializedPropertyAccessException("mService");
}
logTextView.setText((CharSequence)var10001.getOutput());
} else {
Log.w("adx", "readLog mBound==null");
}
}
public final void checkConfig() {
// Intrinsics.checkNotNullExpressionValue(var12, "resources");
AssetManager assetmanager = getResources().getAssets();
FileOutputStream fileOutputStream = null;
try {
fileOutputStream = this.openFileOutput("config.toml", 0);
} catch (FileNotFoundException e) {
e.printStackTrace();
}
if (fileOutputStream == null) {
return;
}
InputStream inputStream = null;
byte[] bytes = new byte[256];
try {
boolean var6 = false;
inputStream = assetmanager.open("config.toml");
// Intrinsics.checkNotNullExpressionValue(var10001, "assetmanager.open((BuildConfig.ConfigFileName))");
int bytesRead = inputStream.read(bytes);
if (bytesRead > 0) {
fileOutputStream.write(bytes, 0, bytesRead);
}
// Unit var11 = Unit.INSTANCE;
} catch (Throwable ex) {
ex.printStackTrace();
} finally {
closeFinally(fileOutputStream);
closeFinally(inputStream);
}
}
private static final void closeFinally(Closeable closable) {
if (closable != null) {
try {
closable.close();
} catch (Throwable var3) {
// ExceptionsKt.addSuppressed(cause, var3);
}
}
}
private final void startShell() {
Intent intent = new Intent((Context)this, ShellService.class);
intent.putExtra("filename", "libfrpc.so");
this.startService(intent);
this.bindService(intent, (ServiceConnection)this.connection, Context.BIND_AUTO_CREATE);
}
private final void stopShell() {
Intent intent = new Intent((Context)this, ShellService.class);
this.unbindService((ServiceConnection)this.connection);
this.stopService(intent);
}
private final void checkNotificationPermission() {
// Intrinsics.checkNotNullExpressionValue(this.registerForActivityResult((ActivityResultContract)(new RequestPermission()), (ActivityResultCallback)null.INSTANCE), "registerForActivityResul….\n }\n }");
}
private final void createBGNotificationChannel() {
if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.O) {
String name = "frp后台服务";
String descriptionText = "frp后台服务通知";
NotificationChannel var5 = null;
var5 = new NotificationChannel("shell_bg", (CharSequence)name, NotificationManager.IMPORTANCE_MIN);
var5.setDescription(descriptionText);
NotificationChannel channel = var5;
NotificationManager notificationManager = (NotificationManager)this.getSystemService(Context.NOTIFICATION_SERVICE);
if (notificationManager == null) {
throw new NullPointerException("null cannot be cast to non-null type android.app.NotificationManager");
}
notificationManager.createNotificationChannel(channel);
}
}
private final boolean isServiceRunning(Class serviceClass) {
ActivityManager activityManager = (ActivityManager)getSystemService(Context.ACTIVITY_SERVICE);
if (activityManager != null) {
Iterator it = activityManager.getRunningServices(Integer.MAX_VALUE).iterator();
ComponentName componentName;
String className;
do {
if (!it.hasNext()) {
return false;
}
ActivityManager.RunningServiceInfo serviceInfo = (ActivityManager.RunningServiceInfo)it.next();
className = serviceClass.getName();
componentName = serviceInfo.service;
// Intrinsics.checkNotNullExpressionValue(var10001, "service.service");
} while(!className.equalsIgnoreCase(componentName.getClassName()));
}
return true;
}
}

@ -0,0 +1,157 @@
package com.xypower.frpandroid;
import android.app.Notification;
import android.app.PendingIntent;
import android.app.Service;
import android.content.Context;
import android.content.Intent;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageManager;
import android.os.Binder;
import android.os.Bundle;
import android.os.IBinder;
import android.support.annotation.Nullable;
import android.util.Log;
import android.widget.Toast;
import org.jetbrains.annotations.NotNull;
import java.io.File;
import java.util.Random;
public class ShellService extends Service {
private Thread process_thread;
private final StringBuilder outputBuilder = new StringBuilder();
private final LocalBinder binder = new LocalBinder();
private final Random mGenerator = new Random();
public ShellService() {
}
@Override
public IBinder onBind(Intent intent) {
// Intrinsics.checkNotNullParameter(intent, "intent");
return (IBinder)this.binder;
}
@NotNull
public final String getOutput() {
return this.outputBuilder.toString();
// Intrinsics.checkNotNullExpressionValue(var10000, "outputBuilder.toString()");
}
public final void clearOutput() {
// StringsKt.clear(this.outputBuilder);
// this.outputBuilder.
this.outputBuilder.delete(0, outputBuilder.length());
}
public final int getRandomNumber() {
return this.mGenerator.nextInt(100);
}
public int onStartCommand(@Nullable Intent intent, int flags, int startId) {
String filename = "";
if (this.process_thread != null) {
Log.w("adx", "process isn't null,service won't start");
Toast.makeText((Context)this, (CharSequence)"process isn't null,service won't start", Toast.LENGTH_SHORT).show();
return Service.START_NOT_STICKY;
} else if (intent != null) {
Bundle var10000 = intent.getExtras();
Object var8 = var10000 != null ? var10000.get("filename") : null;
if (var8 == null) {
throw new NullPointerException("null cannot be cast to non-null type kotlin.String");
} else {
filename = (String)var8;
ApplicationInfo applicationInfo = null;
try {
applicationInfo = this.getPackageManager().getApplicationInfo(this.getPackageName(), PackageManager.GET_SHARED_LIBRARY_FILES);
} catch (Exception ex) {
}
// Intrinsics.checkNotNullExpressionValue(var9, "packageManager.getApplic…GET_SHARED_LIBRARY_FILES)");
Log.d("adx", "native library dir " + applicationInfo.nativeLibraryDir);
try {
String var10001 = applicationInfo.nativeLibraryDir + '/' + filename + " -c config.toml";
String[] var10002 = new String[]{""};
File var10003 = this.getFilesDir();
// Intrinsics.checkNotNullExpressionValue(var10003, "this.filesDir");
this.runCommand(var10001, var10002, var10003);
} catch (Exception ex) {
// Log.e("adx", ExceptionsKt.stackTraceToString((Throwable)var7));
ex.printStackTrace();
Toast.makeText((Context)this, (CharSequence)ex.getMessage(), Toast.LENGTH_SHORT).show();
this.stopSelf();
return Service.START_NOT_STICKY;
}
Toast.makeText((Context)this, (CharSequence)"已启动frp服务", Toast.LENGTH_SHORT).show();
this.startForeground(1, this.showMotification());
return Service.START_NOT_STICKY;
}
} else {
filename = "Error:filename unknown!!!";
Log.e("adx", filename);
Toast.makeText((Context)this, (CharSequence)filename, Toast.LENGTH_SHORT).show();
this.stopSelf();
return Service.START_NOT_STICKY;
}
}
public void onDestroy() {
Thread var10000 = this.process_thread;
if (var10000 != null) {
var10000.interrupt();
}
Toast.makeText((Context)this, (CharSequence)"已关闭frp服务", 0).show();
}
private final void runCommand(String command, String[] envp, File dir) {
this.process_thread = new ShellThread(command, envp, dir, new ShellThread.Function1() {
@Override
public Object invoke(Object var1) {
this.invoke((String)var1);
return null;
// return Unit.INSTANCE;
}
public final void invoke(@NotNull String it) {
// Intrinsics.checkNotNullParameter(it, "it");
ShellService.this.outputBuilder.append(it + "\n");
}
});
Thread var10000 = this.process_thread;
if (var10000 != null) {
var10000.start();
}
}
private final Notification showMotification() {
Intent var2 = new Intent((Context)this, MainActivity.class);
boolean var4 = false;
PendingIntent var10000 = PendingIntent.getActivity((Context)this, 0, var2, PendingIntent.FLAG_IMMUTABLE);
// Intrinsics.checkNotNullExpressionValue(var10000, "Intent(this, MainActivit…_IMMUTABLE)\n }");
PendingIntent pendingIntent = var10000;
Notification.Builder var5 = (new Notification.Builder((Context)this, "shell_bg")).setSmallIcon(R.drawable.ic_launcher_foreground).setContentTitle((CharSequence)"frp后台服务").setContentText((CharSequence)"已启动frp").setContentIntent(pendingIntent);
// Intrinsics.checkNotNullExpressionValue(var5, "NotificationCompat.Build…tentIntent(pendingIntent)");
Notification.Builder notification = var5;
Notification var6 = notification.build();
// Intrinsics.checkNotNullExpressionValue(var6, "notification.build()");
return var6;
}
public final class LocalBinder extends Binder implements IBinder {
@NotNull
public final ShellService getService() {
return ShellService.this;
}
}
}

@ -0,0 +1,85 @@
package com.xypower.frpandroid;
import org.jetbrains.annotations.NotNull;
import java.io.BufferedReader;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.Reader;
public class ShellThread extends Thread {
public interface Function1 {
Object invoke(Object var1);
}
@NotNull
private final String command;
@NotNull
private final String[] envp;
@NotNull
private final File dir;
@NotNull
private final Function1 outputCallback;
public ShellThread(@NotNull String command, @NotNull String[] envp, @NotNull File dir, @NotNull Function1 outputCallback) {
// Intrinsics.checkNotNullParameter(command, "command");
// Intrinsics.checkNotNullParameter(envp, "envp");
// Intrinsics.checkNotNullParameter(dir, "dir");
// Intrinsics.checkNotNullParameter(outputCallback, "outputCallback");
super();
this.command = command;
this.envp = envp;
this.dir = dir;
this.outputCallback = outputCallback;
}
public void run() {
try {
Process process = Runtime.getRuntime().exec(this.command, this.envp, this.dir);
// Intrinsics.checkNotNullExpressionValue(process, "process");
InputStream inputStream = process.getInputStream();
BufferedReader reader = new BufferedReader((Reader)(new InputStreamReader(inputStream)));
while(true) {
String var5 = reader.readLine();
if (var5 == null || this.isInterrupted()) {
reader.close();
process.destroy();
break;
}
if (var5 != null) {
this.outputCallback.invoke(var5);
}
}
} catch (IOException ex) {
ex.printStackTrace();
}
}
@NotNull
public final String getCommand() {
return this.command;
}
@NotNull
public final String[] getEnvp() {
return this.envp;
}
@NotNull
public final File getDir() {
return this.dir;
}
@NotNull
public final Function1 getOutputCallback() {
return this.outputCallback;
}
}

@ -0,0 +1,30 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:aapt="http://schemas.android.com/aapt"
android:width="108dp"
android:height="108dp"
android:viewportWidth="108"
android:viewportHeight="108">
<path android:pathData="M31,63.928c0,0 6.4,-11 12.1,-13.1c7.2,-2.6 26,-1.4 26,-1.4l38.1,38.1L107,108.928l-32,-1L31,63.928z">
<aapt:attr name="android:fillColor">
<gradient
android:endX="85.84757"
android:endY="92.4963"
android:startX="42.9492"
android:startY="49.59793"
android:type="linear">
<item
android:color="#44000000"
android:offset="0.0" />
<item
android:color="#00000000"
android:offset="1.0" />
</gradient>
</aapt:attr>
</path>
<path
android:fillColor="#FFFFFF"
android:fillType="nonZero"
android:pathData="M65.3,45.828l3.8,-6.6c0.2,-0.4 0.1,-0.9 -0.3,-1.1c-0.4,-0.2 -0.9,-0.1 -1.1,0.3l-3.9,6.7c-6.3,-2.8 -13.4,-2.8 -19.7,0l-3.9,-6.7c-0.2,-0.4 -0.7,-0.5 -1.1,-0.3C38.8,38.328 38.7,38.828 38.9,39.228l3.8,6.6C36.2,49.428 31.7,56.028 31,63.928h46C76.3,56.028 71.8,49.428 65.3,45.828zM43.4,57.328c-0.8,0 -1.5,-0.5 -1.8,-1.2c-0.3,-0.7 -0.1,-1.5 0.4,-2.1c0.5,-0.5 1.4,-0.7 2.1,-0.4c0.7,0.3 1.2,1 1.2,1.8C45.3,56.528 44.5,57.328 43.4,57.328L43.4,57.328zM64.6,57.328c-0.8,0 -1.5,-0.5 -1.8,-1.2s-0.1,-1.5 0.4,-2.1c0.5,-0.5 1.4,-0.7 2.1,-0.4c0.7,0.3 1.2,1 1.2,1.8C66.5,56.528 65.6,57.328 64.6,57.328L64.6,57.328z"
android:strokeWidth="1"
android:strokeColor="#00000000" />
</vector>

@ -0,0 +1,170 @@
<?xml version="1.0" encoding="utf-8"?>
<vector xmlns:android="http://schemas.android.com/apk/res/android"
android:width="108dp"
android:height="108dp"
android:viewportWidth="108"
android:viewportHeight="108">
<path
android:fillColor="#3DDC84"
android:pathData="M0,0h108v108h-108z" />
<path
android:fillColor="#00000000"
android:pathData="M9,0L9,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,0L19,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M29,0L29,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M39,0L39,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M49,0L49,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M59,0L59,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M69,0L69,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M79,0L79,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M89,0L89,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M99,0L99,108"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,9L108,9"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,19L108,19"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,29L108,29"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,39L108,39"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,49L108,49"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,59L108,59"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,69L108,69"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,79L108,79"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,89L108,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M0,99L108,99"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,29L89,29"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,39L89,39"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,49L89,49"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,59L89,59"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,69L89,69"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M19,79L89,79"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M29,19L29,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M39,19L39,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M49,19L49,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M59,19L59,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M69,19L69,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
<path
android:fillColor="#00000000"
android:pathData="M79,19L79,89"
android:strokeWidth="0.8"
android:strokeColor="#33FFFFFF" />
</vector>

@ -0,0 +1,39 @@
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context=".ConfigActivity">
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:orientation="horizontal">
<Button
android:id="@+id/saveConfigButton"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="保存" />
<Button
android:id="@+id/dontSaveConfigButton"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="不保存" />
</LinearLayout>
<EditText
android:id="@+id/configEditText"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:ems="10"
android:gravity="start|top"
android:inputType="textMultiLine"
android:textColor="?android:attr/textColorPrimary"
android:textSize="12sp"
android:typeface="monospace" />
</LinearLayout>

@ -0,0 +1,99 @@
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context=".MainActivity">
<ScrollView
android:layout_width="match_parent"
android:layout_height="match_parent">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical" >
<TextView
android:id="@+id/titleText"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="frp for Android"
android:textColor="?android:attr/textColorPrimary"
android:textSize="24sp" />
<Switch
android:id="@+id/state_switch"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="切换开关" />
<Switch
android:id="@+id/auto_start_switch"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="开机自启动" />
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="horizontal">
<Button
android:id="@+id/configButton"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="设置" />
<Button
android:id="@+id/aboutButton"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="关于" />
</LinearLayout>
<Space
android:layout_width="match_parent"
android:layout_height="40sp" />
<TextView
android:id="@+id/textView5"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="frp日志"
android:textColor="?android:attr/textColorPrimary" />
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:orientation="horizontal">
<Button
android:id="@+id/refreshButton"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="刷新" />
<Button
android:id="@+id/deleteButton"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="清除" />
</LinearLayout>
<TextView
android:id="@+id/logTextView"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="无日志"
android:textColor="?android:attr/textColorPrimary"
android:typeface="monospace" />
</LinearLayout>
</ScrollView>
</LinearLayout>

@ -0,0 +1,5 @@
<?xml version="1.0" encoding="utf-8"?>
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
<background android:drawable="@drawable/ic_launcher_background" />
<foreground android:drawable="@drawable/ic_launcher_foreground" />
</adaptive-icon>

@ -0,0 +1,5 @@
<?xml version="1.0" encoding="utf-8"?>
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
<background android:drawable="@drawable/ic_launcher_background" />
<foreground android:drawable="@drawable/ic_launcher_foreground" />
</adaptive-icon>

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.4 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 982 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.7 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.9 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 7.6 KiB

@ -0,0 +1,10 @@
<resources xmlns:tools="http://schemas.android.com/tools">
<!-- Base application theme. -->
<style name="Theme.FrpAndroid" parent="Theme.AppCompat.Light.DarkActionBar">
<!-- Primary brand color. -->
<item name="colorPrimary">@color/purple_200</item>
<item name="colorPrimaryDark">@color/purple_700</item>
<item name="colorAccent">@color/teal_200</item>
<!-- Customize your theme here. -->
</style>
</resources>

@ -0,0 +1,10 @@
<?xml version="1.0" encoding="utf-8"?>
<resources>
<color name="purple_200">#FFBB86FC</color>
<color name="purple_500">#FF6200EE</color>
<color name="purple_700">#FF3700B3</color>
<color name="teal_200">#FF03DAC5</color>
<color name="teal_700">#FF018786</color>
<color name="black">#FF000000</color>
<color name="white">#FFFFFFFF</color>
</resources>

@ -0,0 +1,3 @@
<resources>
<string name="app_name">frpAndroid</string>
</resources>

@ -0,0 +1,10 @@
<resources xmlns:tools="http://schemas.android.com/tools">
<!-- Base application theme. -->
<style name="Theme.FrpAndroid" parent="Theme.AppCompat.Light.DarkActionBar">
<!-- Primary brand color. -->
<item name="colorPrimary">@color/purple_500</item>
<item name="colorPrimaryDark">@color/purple_700</item>
<item name="colorAccent">@color/teal_200</item>
<!-- Customize your theme here. -->
</style>
</resources>

@ -0,0 +1,13 @@
<?xml version="1.0" encoding="utf-8"?><!--
Sample backup rules file; uncomment and customize as necessary.
See https://developer.android.com/guide/topics/data/autobackup
for details.
Note: This file is ignored for devices older that API 31
See https://developer.android.com/about/versions/12/backup-restore
-->
<full-backup-content>
<!--
<include domain="sharedpref" path="."/>
<exclude domain="sharedpref" path="device.xml"/>
-->
</full-backup-content>

@ -0,0 +1,19 @@
<?xml version="1.0" encoding="utf-8"?><!--
Sample data extraction rules file; uncomment and customize as necessary.
See https://developer.android.com/about/versions/12/backup-restore#xml-changes
for details.
-->
<data-extraction-rules>
<cloud-backup>
<!-- TODO: Use <include> and <exclude> to control what is backed up.
<include .../>
<exclude .../>
-->
</cloud-backup>
<!--
<device-transfer>
<include .../>
<exclude .../>
</device-transfer>
-->
</data-extraction-rules>

@ -0,0 +1,9 @@
// Top-level build file where you can add configuration options common to all sub-projects/modules.
plugins {
id 'com.android.application' version '7.2.2' apply false
id 'com.android.library' version '7.2.2' apply false
}
task clean(type: Delete) {
delete rootProject.buildDir
}

@ -0,0 +1,17 @@
# Project-wide Gradle settings.
# IDE (e.g. Android Studio) users:
# Gradle settings configured through the IDE *will override*
# any settings specified in this file.
# For more details on how to configure your build environment visit
# http://www.gradle.org/docs/current/userguide/build_environment.html
# Specifies the JVM arguments used for the daemon process.
# The setting is particularly useful for tweaking memory settings.
org.gradle.jvmargs=-Xmx2048m -Dfile.encoding=UTF-8
# When configured, Gradle will run in incubating parallel mode.
# This option should only be used with decoupled projects. More details, visit
# http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects
# org.gradle.parallel=true
# Enables namespacing of each library's R class so that its R class includes only the
# resources declared in the library itself and none from the library's dependencies,
# thereby reducing the size of the R class for that library
android.nonTransitiveRClass=true

@ -0,0 +1,6 @@
#Mon Jun 24 21:01:31 CST 2024
distributionBase=GRADLE_USER_HOME
distributionUrl=https\://mirrors.cloud.tencent.com/gradle/gradle-7.3.3-bin.zip
distributionPath=wrapper/dists
zipStorePath=wrapper/dists
zipStoreBase=GRADLE_USER_HOME

185
gradlew vendored

@ -0,0 +1,185 @@
#!/usr/bin/env sh
#
# Copyright 2015 the original author or authors.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# https://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
##############################################################################
##
## Gradle start up script for UN*X
##
##############################################################################
# Attempt to set APP_HOME
# Resolve links: $0 may be a link
PRG="$0"
# Need this for relative symlinks.
while [ -h "$PRG" ] ; do
ls=`ls -ld "$PRG"`
link=`expr "$ls" : '.*-> \(.*\)$'`
if expr "$link" : '/.*' > /dev/null; then
PRG="$link"
else
PRG=`dirname "$PRG"`"/$link"
fi
done
SAVED="`pwd`"
cd "`dirname \"$PRG\"`/" >/dev/null
APP_HOME="`pwd -P`"
cd "$SAVED" >/dev/null
APP_NAME="Gradle"
APP_BASE_NAME=`basename "$0"`
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
# Use the maximum available, or set MAX_FD != -1 to use that value.
MAX_FD="maximum"
warn () {
echo "$*"
}
die () {
echo
echo "$*"
echo
exit 1
}
# OS specific support (must be 'true' or 'false').
cygwin=false
msys=false
darwin=false
nonstop=false
case "`uname`" in
CYGWIN* )
cygwin=true
;;
Darwin* )
darwin=true
;;
MINGW* )
msys=true
;;
NONSTOP* )
nonstop=true
;;
esac
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
# Determine the Java command to use to start the JVM.
if [ -n "$JAVA_HOME" ] ; then
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
# IBM's JDK on AIX uses strange locations for the executables
JAVACMD="$JAVA_HOME/jre/sh/java"
else
JAVACMD="$JAVA_HOME/bin/java"
fi
if [ ! -x "$JAVACMD" ] ; then
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
else
JAVACMD="java"
which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
Please set the JAVA_HOME variable in your environment to match the
location of your Java installation."
fi
# Increase the maximum file descriptors if we can.
if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then
MAX_FD_LIMIT=`ulimit -H -n`
if [ $? -eq 0 ] ; then
if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
MAX_FD="$MAX_FD_LIMIT"
fi
ulimit -n $MAX_FD
if [ $? -ne 0 ] ; then
warn "Could not set maximum file descriptor limit: $MAX_FD"
fi
else
warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
fi
fi
# For Darwin, add options to specify how the application appears in the dock
if $darwin; then
GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
fi
# For Cygwin or MSYS, switch paths to Windows format before running java
if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then
APP_HOME=`cygpath --path --mixed "$APP_HOME"`
CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
JAVACMD=`cygpath --unix "$JAVACMD"`
# We build the pattern for arguments to be converted via cygpath
ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
SEP=""
for dir in $ROOTDIRSRAW ; do
ROOTDIRS="$ROOTDIRS$SEP$dir"
SEP="|"
done
OURCYGPATTERN="(^($ROOTDIRS))"
# Add a user-defined pattern to the cygpath arguments
if [ "$GRADLE_CYGPATTERN" != "" ] ; then
OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
fi
# Now convert the arguments - kludge to limit ourselves to /bin/sh
i=0
for arg in "$@" ; do
CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
else
eval `echo args$i`="\"$arg\""
fi
i=`expr $i + 1`
done
case $i in
0) set -- ;;
1) set -- "$args0" ;;
2) set -- "$args0" "$args1" ;;
3) set -- "$args0" "$args1" "$args2" ;;
4) set -- "$args0" "$args1" "$args2" "$args3" ;;
5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
esac
fi
# Escape application args
save () {
for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done
echo " "
}
APP_ARGS=`save "$@"`
# Collect all arguments for the java command, following the shell quoting and substitution rules
eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS"
exec "$JAVACMD" "$@"

89
gradlew.bat vendored

@ -0,0 +1,89 @@
@rem
@rem Copyright 2015 the original author or authors.
@rem
@rem Licensed under the Apache License, Version 2.0 (the "License");
@rem you may not use this file except in compliance with the License.
@rem You may obtain a copy of the License at
@rem
@rem https://www.apache.org/licenses/LICENSE-2.0
@rem
@rem Unless required by applicable law or agreed to in writing, software
@rem distributed under the License is distributed on an "AS IS" BASIS,
@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
@rem See the License for the specific language governing permissions and
@rem limitations under the License.
@rem
@if "%DEBUG%" == "" @echo off
@rem ##########################################################################
@rem
@rem Gradle startup script for Windows
@rem
@rem ##########################################################################
@rem Set local scope for the variables with windows NT shell
if "%OS%"=="Windows_NT" setlocal
set DIRNAME=%~dp0
if "%DIRNAME%" == "" set DIRNAME=.
set APP_BASE_NAME=%~n0
set APP_HOME=%DIRNAME%
@rem Resolve any "." and ".." in APP_HOME to make it shorter.
for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m"
@rem Find java.exe
if defined JAVA_HOME goto findJavaFromJavaHome
set JAVA_EXE=java.exe
%JAVA_EXE% -version >NUL 2>&1
if "%ERRORLEVEL%" == "0" goto execute
echo.
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
echo.
echo Please set the JAVA_HOME variable in your environment to match the
echo location of your Java installation.
goto fail
:findJavaFromJavaHome
set JAVA_HOME=%JAVA_HOME:"=%
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
if exist "%JAVA_EXE%" goto execute
echo.
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
echo.
echo Please set the JAVA_HOME variable in your environment to match the
echo location of your Java installation.
goto fail
:execute
@rem Setup the command line
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
@rem Execute Gradle
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %*
:end
@rem End local scope for the variables with windows NT shell
if "%ERRORLEVEL%"=="0" goto mainEnd
:fail
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
rem the _cmd.exe /c_ return code!
if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
exit /b 1
:mainEnd
if "%OS%"=="Windows_NT" endlocal
:omega

@ -0,0 +1,16 @@
pluginManagement {
repositories {
gradlePluginPortal()
google()
mavenCentral()
}
}
dependencyResolutionManagement {
repositoriesMode.set(RepositoriesMode.FAIL_ON_PROJECT_REPOS)
repositories {
google()
mavenCentral()
}
}
rootProject.name = "frpAndroid"
include ':app'
Loading…
Cancel
Save