2025-07-11 18:31:33 +08:00
|
|
|
|
package com.example.myapplication.Service;
|
|
|
|
|
|
|
|
|
|
import android.annotation.SuppressLint;
|
|
|
|
|
import android.app.Notification;
|
|
|
|
|
import android.app.NotificationChannel;
|
|
|
|
|
import android.app.NotificationManager;
|
|
|
|
|
import android.app.Service;
|
|
|
|
|
import android.content.Context;
|
|
|
|
|
import android.content.Intent;
|
|
|
|
|
import android.content.pm.ServiceInfo;
|
|
|
|
|
import android.graphics.Color;
|
|
|
|
|
import android.graphics.PixelFormat;
|
|
|
|
|
import android.graphics.Rect;
|
|
|
|
|
import android.net.ConnectivityManager;
|
|
|
|
|
import android.net.NetworkCapabilities;
|
|
|
|
|
import android.os.Build;
|
|
|
|
|
import android.os.Handler;
|
|
|
|
|
import android.os.IBinder;
|
|
|
|
|
import android.os.Looper;
|
2025-07-25 14:29:27 +08:00
|
|
|
|
import android.os.VibrationEffect;
|
|
|
|
|
import android.os.Vibrator;
|
2025-07-11 18:31:33 +08:00
|
|
|
|
import android.text.Editable;
|
|
|
|
|
import android.text.TextUtils;
|
|
|
|
|
import android.text.TextWatcher;
|
|
|
|
|
import android.util.Log;
|
|
|
|
|
import android.view.Gravity;
|
|
|
|
|
import android.view.KeyEvent;
|
|
|
|
|
import android.view.LayoutInflater;
|
|
|
|
|
import android.view.MotionEvent;
|
|
|
|
|
import android.view.View;
|
|
|
|
|
import android.view.ViewGroup;
|
|
|
|
|
import android.view.WindowManager;
|
|
|
|
|
import android.view.inputmethod.InputMethodManager;
|
|
|
|
|
import android.widget.ArrayAdapter;
|
|
|
|
|
import android.widget.AutoCompleteTextView;
|
|
|
|
|
import android.widget.Filter;
|
|
|
|
|
import android.widget.RadioButton;
|
|
|
|
|
import android.widget.RadioGroup;
|
|
|
|
|
import android.widget.TextView;
|
|
|
|
|
import android.widget.Toast;
|
|
|
|
|
|
|
|
|
|
import androidx.annotation.NonNull;
|
|
|
|
|
import androidx.annotation.Nullable;
|
|
|
|
|
import androidx.core.app.NotificationCompat;
|
|
|
|
|
|
|
|
|
|
import com.example.myapplication.DataBase.AppDatabase;
|
|
|
|
|
import com.example.myapplication.DataBase.TurbineDao;
|
|
|
|
|
import com.example.myapplication.R;
|
|
|
|
|
import com.example.myapplication.api.TurbineApiService;
|
|
|
|
|
import com.example.myapplication.model.ApiResponse;
|
|
|
|
|
import com.example.myapplication.model.Turbine;
|
|
|
|
|
|
|
|
|
|
import java.io.IOException;
|
|
|
|
|
import java.util.List;
|
|
|
|
|
import java.util.concurrent.ExecutorService;
|
|
|
|
|
import java.util.concurrent.Executors;
|
|
|
|
|
|
|
|
|
|
import retrofit2.Call;
|
|
|
|
|
import retrofit2.Callback;
|
|
|
|
|
import retrofit2.Response;
|
|
|
|
|
import retrofit2.Retrofit;
|
|
|
|
|
import retrofit2.converter.gson.GsonConverterFactory;
|
|
|
|
|
|
|
|
|
|
public class FloatingWindowService extends Service {
|
|
|
|
|
private static final String CHANNEL_ID = "FloatingWindowChannel";
|
|
|
|
|
private static final int NOTIFICATION_ID = 1;
|
|
|
|
|
private Handler mainHandler = new Handler(Looper.getMainLooper());
|
|
|
|
|
|
|
|
|
|
private WindowManager windowManager;
|
|
|
|
|
private View floatingView;
|
|
|
|
|
|
|
|
|
|
private RadioGroup radioGroup;
|
|
|
|
|
private static final String EXTRA_PROJECT_ID = "PROJECT_ID";
|
|
|
|
|
private String lastUnit = "";
|
|
|
|
|
private int lastBlade = -1;
|
|
|
|
|
private String lastUnitName = "";
|
|
|
|
|
private AppDatabase database;
|
|
|
|
|
private TurbineDao turbineDao;
|
2025-07-25 14:29:27 +08:00
|
|
|
|
private AutoCompleteTextView actvUnit;
|
2025-07-11 18:31:33 +08:00
|
|
|
|
private ArrayAdapter<Turbine> turbineAdapter;
|
|
|
|
|
private String projectID;
|
2025-07-25 14:29:27 +08:00
|
|
|
|
public static boolean isFloatingWindowShowing = false;
|
|
|
|
|
|
2025-07-11 18:31:33 +08:00
|
|
|
|
@SuppressLint("SetTextI18n")
|
|
|
|
|
@Override
|
|
|
|
|
public void onCreate() {
|
|
|
|
|
super.onCreate();
|
|
|
|
|
createNotificationChannel();
|
|
|
|
|
database = AppDatabase.getDatabase(this);
|
|
|
|
|
turbineDao = database.turbineDao();
|
|
|
|
|
showFloatingWindow();
|
|
|
|
|
}
|
2025-07-25 14:29:27 +08:00
|
|
|
|
|
2025-07-11 18:31:33 +08:00
|
|
|
|
@Override
|
|
|
|
|
public int onStartCommand(Intent intent, int flags, int startId) {
|
|
|
|
|
if (intent != null && intent.hasExtra(EXTRA_PROJECT_ID)) {
|
|
|
|
|
projectID = intent.getStringExtra(EXTRA_PROJECT_ID);
|
|
|
|
|
}
|
|
|
|
|
return super.onStartCommand(intent, flags, startId);
|
|
|
|
|
}
|
2025-07-25 14:29:27 +08:00
|
|
|
|
|
2025-07-11 18:31:33 +08:00
|
|
|
|
private void createNotificationChannel() {
|
|
|
|
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
|
|
|
|
|
NotificationChannel channel = new NotificationChannel(
|
|
|
|
|
CHANNEL_ID,
|
|
|
|
|
"悬浮窗服务通道",
|
|
|
|
|
NotificationManager.IMPORTANCE_LOW
|
|
|
|
|
);
|
|
|
|
|
NotificationManager manager = getSystemService(NotificationManager.class);
|
|
|
|
|
manager.createNotificationChannel(channel);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void showFloatingWindow() {
|
|
|
|
|
// 创建前台服务通知
|
2025-07-25 14:29:27 +08:00
|
|
|
|
isFloatingWindowShowing = true;
|
2025-07-11 18:31:33 +08:00
|
|
|
|
Notification notification = new NotificationCompat.Builder(this, CHANNEL_ID)
|
|
|
|
|
.setContentTitle("悬浮窗服务运行中")
|
|
|
|
|
.setContentText("正在显示悬浮提示框")
|
|
|
|
|
.setSmallIcon(R.mipmap.ic_launcher)
|
|
|
|
|
.setPriority(NotificationCompat.PRIORITY_LOW)
|
|
|
|
|
.build();
|
|
|
|
|
|
|
|
|
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
|
|
|
|
|
startForeground(NOTIFICATION_ID, notification,
|
|
|
|
|
ServiceInfo.FOREGROUND_SERVICE_TYPE_SPECIAL_USE);
|
|
|
|
|
} else {
|
|
|
|
|
startForeground(NOTIFICATION_ID, notification);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 创建悬浮窗视图
|
|
|
|
|
floatingView = LayoutInflater.from(this).inflate(R.layout.floating_window, null);
|
|
|
|
|
|
|
|
|
|
// 设置窗口参数
|
|
|
|
|
final WindowManager.LayoutParams params = new WindowManager.LayoutParams(
|
|
|
|
|
WindowManager.LayoutParams.WRAP_CONTENT,
|
|
|
|
|
WindowManager.LayoutParams.WRAP_CONTENT,
|
|
|
|
|
Build.VERSION.SDK_INT >= Build.VERSION_CODES.O ?
|
|
|
|
|
WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY :
|
|
|
|
|
WindowManager.LayoutParams.TYPE_PHONE,
|
|
|
|
|
WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL |
|
|
|
|
|
WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH,
|
|
|
|
|
PixelFormat.TRANSLUCENT);
|
|
|
|
|
|
|
|
|
|
params.gravity = Gravity.TOP | Gravity.START;
|
|
|
|
|
params.x = 100;
|
|
|
|
|
params.y = 100;
|
|
|
|
|
|
|
|
|
|
windowManager = (WindowManager) getSystemService(WINDOW_SERVICE);
|
|
|
|
|
windowManager.addView(floatingView, params);
|
|
|
|
|
|
|
|
|
|
// 初始化视图
|
|
|
|
|
actvUnit = floatingView.findViewById(R.id.actvUnit);
|
|
|
|
|
setupTurbineDropdown();
|
|
|
|
|
radioGroup = floatingView.findViewById(R.id.radioGroup);
|
|
|
|
|
|
2025-07-25 14:29:27 +08:00
|
|
|
|
// 设置关闭按钮点击事件
|
|
|
|
|
View btnClose = floatingView.findViewById(R.id.btnClose);
|
|
|
|
|
btnClose.setOnClickListener(v -> {
|
|
|
|
|
// 震动反馈
|
|
|
|
|
Vibrator vibrator = (Vibrator) getSystemService(VIBRATOR_SERVICE);
|
|
|
|
|
if (vibrator != null && vibrator.hasVibrator()) {
|
|
|
|
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
|
|
|
|
|
vibrator.vibrate(VibrationEffect.createOneShot(50, VibrationEffect.DEFAULT_AMPLITUDE));
|
|
|
|
|
} else {
|
|
|
|
|
vibrator.vibrate(50);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
closeFloatingWindow();
|
|
|
|
|
});
|
|
|
|
|
|
2025-07-11 18:31:33 +08:00
|
|
|
|
setupDragListener(params);
|
|
|
|
|
setupValueChangeListeners();
|
|
|
|
|
setupAutoCompleteBehavior();
|
2025-07-25 14:29:27 +08:00
|
|
|
|
}
|
2025-07-11 18:31:33 +08:00
|
|
|
|
|
2025-07-25 14:29:27 +08:00
|
|
|
|
private void closeFloatingWindow() {
|
|
|
|
|
if (floatingView != null && floatingView.getWindowToken() != null) {
|
|
|
|
|
// 添加淡出动画
|
|
|
|
|
floatingView.animate()
|
|
|
|
|
.alpha(0f)
|
|
|
|
|
.setDuration(200)
|
|
|
|
|
.withEndAction(() -> {
|
|
|
|
|
removeFloatingView();
|
|
|
|
|
stopSelf();
|
|
|
|
|
})
|
|
|
|
|
.start();
|
|
|
|
|
} else {
|
|
|
|
|
removeFloatingView();
|
|
|
|
|
stopSelf();
|
|
|
|
|
}
|
|
|
|
|
isFloatingWindowShowing = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private synchronized void removeFloatingView() {
|
|
|
|
|
try {
|
|
|
|
|
if (windowManager != null && floatingView != null && floatingView.getWindowToken() != null) {
|
|
|
|
|
windowManager.removeView(floatingView);
|
|
|
|
|
}
|
|
|
|
|
} catch (IllegalArgumentException e) {
|
|
|
|
|
Log.e("FloatingWindow", "View already removed", e);
|
|
|
|
|
} finally {
|
|
|
|
|
floatingView = null;
|
|
|
|
|
}
|
2025-07-11 18:31:33 +08:00
|
|
|
|
}
|
|
|
|
|
private void setupAutoCompleteBehavior() {
|
2025-07-25 14:29:27 +08:00
|
|
|
|
actvUnit.setFocusableInTouchMode(true);
|
2025-07-11 18:31:33 +08:00
|
|
|
|
actvUnit.setFocusable(true);
|
|
|
|
|
actvUnit.setOnClickListener(v -> {
|
|
|
|
|
if (turbineAdapter.getCount() > 0) {
|
|
|
|
|
actvUnit.showDropDown();
|
|
|
|
|
} else {
|
|
|
|
|
actvUnit.requestFocus();
|
|
|
|
|
showSoftInputDelayed(actvUnit);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
actvUnit.setOnFocusChangeListener((v, hasFocus) -> {
|
|
|
|
|
if (hasFocus && !TextUtils.isEmpty(actvUnit.getText()) && turbineAdapter.getCount() > 0) {
|
|
|
|
|
actvUnit.postDelayed(() -> actvUnit.showDropDown(), 100);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
actvUnit.setOnKeyListener((v, keyCode, event) -> {
|
|
|
|
|
if (keyCode == KeyEvent.KEYCODE_ENTER && event.getAction() == KeyEvent.ACTION_UP) {
|
|
|
|
|
if (turbineAdapter.getCount() > 0) {
|
|
|
|
|
actvUnit.showDropDown();
|
|
|
|
|
} else {
|
|
|
|
|
hideSoftInput(actvUnit);
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
});
|
|
|
|
|
}
|
2025-07-25 14:29:27 +08:00
|
|
|
|
|
2025-07-11 18:31:33 +08:00
|
|
|
|
private void showSoftInputDelayed(View view) {
|
|
|
|
|
view.postDelayed(() -> {
|
|
|
|
|
InputMethodManager imm = (InputMethodManager) getSystemService(INPUT_METHOD_SERVICE);
|
|
|
|
|
if (imm != null) {
|
|
|
|
|
imm.showSoftInput(view, InputMethodManager.SHOW_IMPLICIT);
|
|
|
|
|
}
|
|
|
|
|
}, 200);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void hideSoftInput(View view) {
|
|
|
|
|
InputMethodManager imm = (InputMethodManager) getSystemService(INPUT_METHOD_SERVICE);
|
|
|
|
|
if (imm != null) {
|
|
|
|
|
imm.hideSoftInputFromWindow(view.getWindowToken(), 0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void setupDragListener(WindowManager.LayoutParams params) {
|
2025-07-25 14:29:27 +08:00
|
|
|
|
View dragHandle = floatingView.findViewById(R.id.floatingContainer);
|
|
|
|
|
|
|
|
|
|
dragHandle.setOnTouchListener(new View.OnTouchListener() {
|
2025-07-11 18:31:33 +08:00
|
|
|
|
private int initialX, initialY;
|
|
|
|
|
private float initialTouchX, initialTouchY;
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public boolean onTouch(View v, MotionEvent event) {
|
2025-07-25 14:29:27 +08:00
|
|
|
|
// 检查是否点击在关闭按钮上
|
|
|
|
|
if (!isViewAttached()) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
Rect closeButtonRect = new Rect();
|
|
|
|
|
floatingView.findViewById(R.id.btnClose).getGlobalVisibleRect(closeButtonRect);
|
|
|
|
|
if (closeButtonRect.contains((int)event.getRawX(), (int)event.getRawY())) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 检查是否点击在输入区域
|
|
|
|
|
Rect inputRect = new Rect();
|
|
|
|
|
actvUnit.getGlobalVisibleRect(inputRect);
|
|
|
|
|
if (inputRect.contains((int)event.getRawX(), (int)event.getRawY())) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-11 18:31:33 +08:00
|
|
|
|
switch (event.getAction()) {
|
|
|
|
|
case MotionEvent.ACTION_DOWN:
|
|
|
|
|
initialX = params.x;
|
|
|
|
|
initialY = params.y;
|
|
|
|
|
initialTouchX = event.getRawX();
|
|
|
|
|
initialTouchY = event.getRawY();
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
case MotionEvent.ACTION_MOVE:
|
|
|
|
|
params.x = initialX + (int)(event.getRawX() - initialTouchX);
|
|
|
|
|
params.y = initialY + (int)(event.getRawY() - initialTouchY);
|
|
|
|
|
windowManager.updateViewLayout(floatingView, params);
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
case MotionEvent.ACTION_UP:
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
2025-07-25 14:29:27 +08:00
|
|
|
|
|
2025-07-11 18:31:33 +08:00
|
|
|
|
private void setupTurbineDropdown() {
|
|
|
|
|
turbineAdapter = new ArrayAdapter<Turbine>(this,
|
|
|
|
|
R.layout.dropdown_item, R.id.text1) {
|
|
|
|
|
@NonNull
|
|
|
|
|
@Override
|
|
|
|
|
public View getView(int position, @Nullable View convertView, @NonNull ViewGroup parent) {
|
|
|
|
|
View view = super.getView(position, convertView, parent);
|
|
|
|
|
TextView textView = view.findViewById(R.id.text1);
|
|
|
|
|
Turbine turbine = getItem(position);
|
|
|
|
|
if (turbine != null) {
|
|
|
|
|
textView.setText(String.format(turbine.turbineName));
|
|
|
|
|
textView.setSingleLine(false);
|
|
|
|
|
textView.setMaxLines(3);
|
|
|
|
|
}
|
|
|
|
|
return view;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public Filter getFilter() {
|
|
|
|
|
return new Filter() {
|
|
|
|
|
@Override
|
|
|
|
|
protected FilterResults performFiltering(CharSequence constraint) {
|
|
|
|
|
FilterResults results = new FilterResults();
|
|
|
|
|
List<Turbine> filtered;
|
|
|
|
|
if (TextUtils.isEmpty(constraint)) {
|
|
|
|
|
filtered = turbineDao.getTurbinesByProject(projectID);
|
|
|
|
|
} else {
|
|
|
|
|
filtered = turbineDao.searchTurbines(projectID, "%" + constraint + "%");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
results.values = filtered;
|
|
|
|
|
results.count = filtered.size();
|
|
|
|
|
return results;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
protected void publishResults(CharSequence constraint, FilterResults results) {
|
|
|
|
|
clear();
|
|
|
|
|
if (results.count > 0) {
|
|
|
|
|
addAll((List<Turbine>) results.values);
|
|
|
|
|
notifyDataSetChanged();
|
|
|
|
|
if (actvUnit.hasFocus() && !TextUtils.isEmpty(constraint)) {
|
|
|
|
|
actvUnit.post(() -> actvUnit.showDropDown());
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
notifyDataSetInvalidated();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
actvUnit.setAdapter(turbineAdapter);
|
2025-07-25 14:29:27 +08:00
|
|
|
|
actvUnit.setThreshold(0);
|
2025-07-11 18:31:33 +08:00
|
|
|
|
actvUnit.setDropDownHeight(ViewGroup.LayoutParams.WRAP_CONTENT);
|
2025-07-25 14:29:27 +08:00
|
|
|
|
actvUnit.setDropDownVerticalOffset(10);
|
2025-07-11 18:31:33 +08:00
|
|
|
|
|
|
|
|
|
actvUnit.addTextChangedListener(new TextWatcher() {
|
|
|
|
|
@Override public void beforeTextChanged(CharSequence s, int start, int count, int after) {}
|
|
|
|
|
@Override public void afterTextChanged(Editable s) {}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public void onTextChanged(CharSequence s, int start, int before, int count) {
|
|
|
|
|
String input=s.toString().trim();
|
|
|
|
|
lastUnit = input;
|
2025-07-25 14:29:27 +08:00
|
|
|
|
lastUnitName = input;
|
2025-07-11 18:31:33 +08:00
|
|
|
|
if (turbineAdapter.getCount() > 0 && !TextUtils.isEmpty(input)) {
|
|
|
|
|
for (int i = 0; i < turbineAdapter.getCount(); i++) {
|
|
|
|
|
Turbine turbine = turbineAdapter.getItem(i);
|
|
|
|
|
if (turbine != null && turbine.turbineName.equalsIgnoreCase(input)) {
|
|
|
|
|
lastUnit = turbine.turbineId;
|
|
|
|
|
lastUnitName = turbine.turbineName;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-07-25 14:29:27 +08:00
|
|
|
|
checkAndSendUpdate();
|
2025-07-11 18:31:33 +08:00
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
loadTurbines();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void loadTurbines() {
|
|
|
|
|
ExecutorService executor = Executors.newSingleThreadExecutor();
|
|
|
|
|
executor.execute(() -> {
|
|
|
|
|
try {
|
|
|
|
|
List<Turbine> localTurbines = turbineDao.getTurbinesByProject(projectID);
|
|
|
|
|
|
|
|
|
|
if (localTurbines.isEmpty() || isNetworkAvailable()) {
|
|
|
|
|
fetchTurbinesFromApi();
|
|
|
|
|
} else {
|
|
|
|
|
mainHandler.post(() -> updateTurbineList(localTurbines));
|
|
|
|
|
}
|
|
|
|
|
} catch (Exception e) {
|
|
|
|
|
Log.e("TurbineLoad", "加载机组数据失败", e);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
2025-07-25 14:29:27 +08:00
|
|
|
|
|
2025-07-11 18:31:33 +08:00
|
|
|
|
private void updateTurbineList(List<Turbine> turbines) {
|
|
|
|
|
turbineAdapter.clear();
|
|
|
|
|
if (!turbines.isEmpty()) {
|
|
|
|
|
turbineAdapter.addAll(turbines);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!TextUtils.isEmpty(actvUnit.getText())) {
|
|
|
|
|
turbineAdapter.getFilter().filter(actvUnit.getText());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private boolean isNetworkAvailable() {
|
|
|
|
|
ConnectivityManager cm = (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
|
|
|
|
|
if (cm == null) return false;
|
|
|
|
|
|
|
|
|
|
NetworkCapabilities capabilities = cm.getNetworkCapabilities(cm.getActiveNetwork());
|
|
|
|
|
return capabilities != null &&
|
|
|
|
|
(capabilities.hasTransport(NetworkCapabilities.TRANSPORT_WIFI) ||
|
|
|
|
|
capabilities.hasTransport(NetworkCapabilities.TRANSPORT_CELLULAR) ||
|
|
|
|
|
capabilities.hasTransport(NetworkCapabilities.TRANSPORT_ETHERNET));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void fetchTurbinesFromApi() {
|
|
|
|
|
Retrofit retrofit = new Retrofit.Builder()
|
|
|
|
|
.baseUrl("http://pms.dtyx.net:9158/")
|
|
|
|
|
.addConverterFactory(GsonConverterFactory.create())
|
|
|
|
|
.build();
|
|
|
|
|
|
|
|
|
|
TurbineApiService service = retrofit.create(TurbineApiService.class);
|
|
|
|
|
Call<ApiResponse<List<Turbine>>> call = service.getTurbineList(projectID);
|
|
|
|
|
|
|
|
|
|
call.enqueue(new Callback<ApiResponse<List<Turbine>>>() {
|
|
|
|
|
@Override
|
|
|
|
|
public void onResponse(Call<ApiResponse<List<Turbine>>> call, Response<ApiResponse<List<Turbine>>> response) {
|
|
|
|
|
if (response.isSuccessful() && response.body() != null) {
|
|
|
|
|
ApiResponse<List<Turbine>> apiResponse = response.body();
|
|
|
|
|
|
|
|
|
|
if (apiResponse.getCode() == 200 && apiResponse.getData() != null) {
|
|
|
|
|
List<Turbine> turbines = apiResponse.getData();
|
|
|
|
|
Log.d("API_SUCCESS", "Fetched " + turbines.size() + " turbines");
|
|
|
|
|
|
|
|
|
|
new Thread(() -> {
|
|
|
|
|
try {
|
|
|
|
|
turbineDao.deleteByProjectId(projectID);
|
|
|
|
|
for (Turbine turbine : turbines) {
|
2025-07-25 14:29:27 +08:00
|
|
|
|
turbine.projectId = projectID;
|
2025-07-11 18:31:33 +08:00
|
|
|
|
}
|
|
|
|
|
turbineDao.insertAll(turbines);
|
|
|
|
|
|
|
|
|
|
mainHandler.post(() -> {
|
|
|
|
|
updateTurbineList(turbines);
|
|
|
|
|
showToast("数据加载成功");
|
|
|
|
|
});
|
|
|
|
|
} catch (Exception e) {
|
|
|
|
|
Log.e("DB_ERROR", "保存数据失败", e);
|
|
|
|
|
mainHandler.post(() -> showToast("保存数据失败: " + e.getMessage()));
|
|
|
|
|
}
|
|
|
|
|
}).start();
|
|
|
|
|
} else {
|
|
|
|
|
String errorMsg = "服务器错误: " + apiResponse.getCode() + " - " + apiResponse.getMsg();
|
|
|
|
|
Log.e("API_ERROR", errorMsg);
|
|
|
|
|
mainHandler.post(() -> showToast(errorMsg));
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
String errorMsg = "请求失败: HTTP " + response.code();
|
|
|
|
|
if (response.errorBody() != null) {
|
|
|
|
|
try {
|
|
|
|
|
errorMsg += "\n" + response.errorBody().string();
|
|
|
|
|
} catch (IOException e) {
|
|
|
|
|
Log.e("API_ERROR", "解析错误信息失败", e);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Log.e("API_ERROR", errorMsg);
|
|
|
|
|
String finalErrorMsg = errorMsg;
|
|
|
|
|
mainHandler.post(() -> showToast(finalErrorMsg));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public void onFailure(Call<ApiResponse<List<Turbine>>> call, Throwable t) {
|
|
|
|
|
String errorMsg = "网络错误: " + t.getMessage();
|
|
|
|
|
Log.e("NETWORK_ERROR", errorMsg, t);
|
|
|
|
|
mainHandler.post(() -> showToast(errorMsg));
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void showToast(String message) {
|
|
|
|
|
Toast.makeText(this, message, Toast.LENGTH_LONG).show();
|
|
|
|
|
}
|
2025-07-25 14:29:27 +08:00
|
|
|
|
|
2025-07-11 18:31:33 +08:00
|
|
|
|
private void setupValueChangeListeners() {
|
|
|
|
|
actvUnit.addTextChangedListener(new TextWatcher() {
|
|
|
|
|
@Override
|
|
|
|
|
public void beforeTextChanged(CharSequence s, int start, int count, int after) {}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public void onTextChanged(CharSequence s, int start, int before, int count) {}
|
|
|
|
|
|
|
|
|
|
@Override
|
2025-07-25 14:29:27 +08:00
|
|
|
|
public void afterTextChanged(Editable s) {}
|
2025-07-11 18:31:33 +08:00
|
|
|
|
});
|
2025-07-25 14:29:27 +08:00
|
|
|
|
|
2025-07-11 18:31:33 +08:00
|
|
|
|
actvUnit.setOnItemClickListener((parent, view, position, id) -> {
|
|
|
|
|
Turbine selected = turbineAdapter.getItem(position);
|
|
|
|
|
if (selected != null) {
|
|
|
|
|
lastUnit = selected.turbineId;
|
|
|
|
|
lastUnitName = selected.turbineName;
|
2025-07-25 14:29:27 +08:00
|
|
|
|
actvUnit.setText(selected.turbineName);
|
2025-07-11 18:31:33 +08:00
|
|
|
|
checkAndSendUpdate();
|
|
|
|
|
}
|
|
|
|
|
});
|
2025-07-25 14:29:27 +08:00
|
|
|
|
|
2025-07-11 18:31:33 +08:00
|
|
|
|
radioGroup.setOnCheckedChangeListener((group, checkedId) -> {
|
|
|
|
|
int currentBlade = 0;
|
|
|
|
|
if (checkedId != -1) {
|
|
|
|
|
RadioButton radioButton = floatingView.findViewById(checkedId);
|
|
|
|
|
currentBlade = Integer.parseInt(radioButton.getText().toString());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (currentBlade != lastBlade) {
|
|
|
|
|
lastBlade = currentBlade;
|
|
|
|
|
checkAndSendUpdate();
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void checkAndSendUpdate() {
|
|
|
|
|
if (!lastUnit.isEmpty() && lastBlade != -1) {
|
|
|
|
|
sendUpdateBroadcast();
|
|
|
|
|
highlightChanges();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void sendUpdateBroadcast() {
|
|
|
|
|
Intent intent = new Intent("com.example.myapplication.UPDATE_VALUES");
|
|
|
|
|
intent.putExtra("unit", lastUnit);
|
|
|
|
|
intent.putExtra("blade", lastBlade);
|
2025-07-25 14:29:27 +08:00
|
|
|
|
intent.putExtra("unitName", lastUnitName);
|
2025-07-11 18:31:33 +08:00
|
|
|
|
intent.setPackage(getPackageName());
|
|
|
|
|
sendBroadcast(intent);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void highlightChanges() {
|
|
|
|
|
int originalColor = actvUnit.getSolidColor();
|
|
|
|
|
actvUnit.setBackgroundColor(Color.YELLOW);
|
|
|
|
|
radioGroup.setBackgroundColor(Color.YELLOW);
|
|
|
|
|
|
|
|
|
|
new Handler().postDelayed(() -> {
|
|
|
|
|
actvUnit.setBackgroundColor(originalColor);
|
|
|
|
|
radioGroup.setBackgroundColor(Color.TRANSPARENT);
|
2025-07-25 14:29:27 +08:00
|
|
|
|
}, 300);
|
2025-07-11 18:31:33 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
public void onDestroy() {
|
|
|
|
|
super.onDestroy();
|
|
|
|
|
hideSoftInput(actvUnit);
|
|
|
|
|
actvUnit.dismissDropDown();
|
2025-07-25 14:29:27 +08:00
|
|
|
|
isFloatingWindowShowing = false;
|
|
|
|
|
|
|
|
|
|
// 直接调用移除方法,不再重复执行stopSelf()
|
|
|
|
|
removeFloatingView();
|
|
|
|
|
|
2025-07-11 18:31:33 +08:00
|
|
|
|
mainHandler.removeCallbacksAndMessages(null);
|
|
|
|
|
}
|
2025-07-25 14:29:27 +08:00
|
|
|
|
private boolean isViewAttached() {
|
|
|
|
|
return floatingView != null
|
|
|
|
|
&& floatingView.getWindowToken() != null
|
|
|
|
|
&& windowManager != null;
|
|
|
|
|
}
|
2025-07-11 18:31:33 +08:00
|
|
|
|
@Override
|
|
|
|
|
public IBinder onBind(Intent intent) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
2025-07-25 14:29:27 +08:00
|
|
|
|
|
|
|
|
|
public static void closeFloatingWindow(Context context) {
|
|
|
|
|
isFloatingWindowShowing = false;
|
|
|
|
|
Intent serviceIntent = new Intent(context, FloatingWindowService.class);
|
|
|
|
|
context.stopService(serviceIntent);
|
|
|
|
|
}
|
2025-07-11 18:31:33 +08:00
|
|
|
|
}
|