Initial commit

master
saad.siddiq 2026-01-27 10:08:35 +05:00
commit eb02e85eb9
199 changed files with 14538 additions and 0 deletions

15
.gitignore vendored Normal file
View File

@ -0,0 +1,15 @@
*.iml
.gradle
/local.properties
/.idea/caches
/.idea/libraries
/.idea/modules.xml
/.idea/workspace.xml
/.idea/navEditor.xml
/.idea/assetWizardSettings.xml
.DS_Store
/build
/captures
.externalNativeBuild
.cxx
local.properties

3
.idea/.gitignore vendored Normal file
View File

@ -0,0 +1,3 @@
# Default ignored files
/shelf/
/workspace.xml

1
.idea/.name Normal file
View File

@ -0,0 +1 @@
QC-Android-App

6
.idea/compiler.xml Normal file
View File

@ -0,0 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="CompilerConfiguration">
<bytecodeTargetLevel target="17" />
</component>
</project>

View File

@ -0,0 +1,10 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="deploymentTargetSelector">
<selectionStates>
<SelectionState runConfigName="app">
<option name="selectionMode" value="DROPDOWN" />
</SelectionState>
</selectionStates>
</component>
</project>

19
.idea/gradle.xml Normal file
View File

@ -0,0 +1,19 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="GradleMigrationSettings" migrationVersion="1" />
<component name="GradleSettings">
<option name="linkedExternalProjectsSettings">
<GradleProjectSettings>
<option name="externalProjectPath" value="$PROJECT_DIR$" />
<option name="gradleJvm" value="#GRADLE_LOCAL_JAVA_HOME" />
<option name="modules">
<set>
<option value="$PROJECT_DIR$" />
<option value="$PROJECT_DIR$/app" />
</set>
</option>
<option name="resolveExternalAnnotations" value="false" />
</GradleProjectSettings>
</option>
</component>
</project>

10
.idea/migrations.xml Normal file
View File

@ -0,0 +1,10 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="ProjectMigrations">
<option name="MigrateToGradleLocalJavaHome">
<set>
<option value="$PROJECT_DIR$" />
</set>
</option>
</component>
</project>

9
.idea/misc.xml Normal file
View File

@ -0,0 +1,9 @@
<project version="4">
<component name="ExternalStorageConfigurationManager" enabled="true" />
<component name="ProjectRootManager" version="2" languageLevel="JDK_17" default="true" project-jdk-name="jbr-17" project-jdk-type="JavaSDK">
<output url="file://$PROJECT_DIR$/build/classes" />
</component>
<component name="ProjectType">
<option name="id" value="Android" />
</component>
</project>

6
.idea/vcs.xml Normal file
View File

@ -0,0 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="VcsDirectoryMappings">
<mapping directory="$PROJECT_DIR$" vcs="Git" />
</component>
</project>

7
app/.gitignore vendored Normal file
View File

@ -0,0 +1,7 @@
*.iml
.gradle
/local.properties
/.idea
/build
/captures
.externalNativeBuild

57
app/build.gradle Normal file
View File

@ -0,0 +1,57 @@
plugins {
alias(libs.plugins.android.application)
}
android {
namespace 'com.utopiaindustries.qc_android'
compileSdk 34
defaultConfig {
applicationId "com.utopiaindustries.qc_android"
minSdk 24
targetSdk 34
versionCode 1
versionName "1.0"
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
}
}
compileOptions {
sourceCompatibility JavaVersion.VERSION_1_8
targetCompatibility JavaVersion.VERSION_1_8
}
}
dependencies {
implementation libs.appcompat
implementation libs.material
implementation libs.activity
implementation libs.constraintlayout
testImplementation libs.junit
androidTestImplementation libs.ext.junit
androidTestImplementation libs.espresso.core
implementation libs.retrofit
implementation libs.converter.gson
implementation libs.glide
implementation libs.easypermissions
implementation libs.preference
implementation 'com.github.MikeOrtiz:TouchImageView:3.6'
implementation 'com.fasterxml.jackson.core:jackson-databind:2.0.1'
implementation 'com.squareup.retrofit2:converter-jackson:2.9.0'
coreLibraryDesugaring 'com.android.tools:desugar_jdk_libs:2.0.3'
implementation 'androidx.navigation:navigation-fragment:2.4.0'
implementation 'androidx.navigation:navigation-ui:2.4.0'
implementation(libs.room.runtime)
annotationProcessor(libs.room.compiler)
}

21
app/proguard-rules.pro vendored Normal file
View File

@ -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

View File

@ -0,0 +1,26 @@
package com.utopiaindustries.qc_android;
import android.content.Context;
import androidx.test.platform.app.InstrumentationRegistry;
import androidx.test.ext.junit.runners.AndroidJUnit4;
import org.junit.Test;
import org.junit.runner.RunWith;
import static org.junit.Assert.*;
/**
* Instrumented test, which will execute on an Android device.
*
* @see <a href="http://d.android.com/tools/testing">Testing documentation</a>
*/
@RunWith(AndroidJUnit4.class)
public class ExampleInstrumentedTest {
@Test
public void useAppContext() {
// Context of the app under test.
Context appContext = InstrumentationRegistry.getInstrumentation().getTargetContext();
assertEquals("com.utopiaindustries.qc_android", appContext.getPackageName());
}
}

View File

@ -0,0 +1,67 @@
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools">
<uses-feature
android:name="android.hardware.camera"
android:required="false" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.CAMERA" />
<uses-permission
android:name="android.permission.READ_EXTERNAL_STORAGE"
android:maxSdkVersion="32" />
<uses-permission
android:name="android.permission.READ_MEDIA_IMAGES"
tools:ignore="SelectedPhotoAccess" />
<uses-permission
android:name="android.permission.WRITE_EXTERNAL_STORAGE"
tools:ignore="ScopedStorage" />
<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:networkSecurityConfig="@xml/network_security_config"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/Theme.QCAndroidApp"
android:usesCleartextTraffic="true"
tools:ignore="DiscouragedApi,LockedOrientationActivity"
tools:targetApi="31">
<activity
android:name=".activities.QCTerryActivity"
android:screenOrientation="portrait"
android:exported="false" />
<activity
android:name=".activities.LoginActivity"
android:exported="true"
android:screenOrientation="portrait"
android:windowSoftInputMode="stateHidden|adjustPan">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity
android:name=".activities.DashboardActivity"
android:exported="true"
android:screenOrientation="portrait"
android:windowSoftInputMode="stateHidden|adjustPan"/>
<provider
android:name="androidx.core.content.FileProvider"
android:authorities="com.utopiaindustries.qc_android"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths" />
</provider>
</application>
</manifest>

View File

@ -0,0 +1,175 @@
package com.utopiaindustries.qc_android.activities;
import android.content.Context;
import android.content.Intent;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.net.ConnectivityManager;
import android.net.NetworkInfo;
import android.os.Bundle;
import android.util.DisplayMetrics;
import android.util.Log;
import android.view.Window;
import android.view.WindowManager;
import android.widget.Button;
import android.widget.Toast;
import androidx.activity.EdgeToEdge;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.content.ContextCompat;
import androidx.core.graphics.Insets;
import androidx.core.view.ViewCompat;
import androidx.core.view.WindowCompat;
import androidx.core.view.WindowInsetsCompat;
import androidx.core.view.WindowInsetsControllerCompat;
import androidx.lifecycle.ViewModelProvider;
import androidx.viewpager.widget.ViewPager;
import com.utopiaindustries.qc_android.R;
import com.utopiaindustries.qc_android.adapters.PagerAdapter;
import com.google.android.material.tabs.TabLayout;
import com.utopiaindustries.qc_android.helper.Helper;
import com.utopiaindustries.qc_android.utils.ProgressDialogFragment;
import com.utopiaindustries.qc_android.viewmodels.ServiceViewModel;
public class DashboardActivity extends AppCompatActivity {
Button btnCreateReport, btnFetchProduct, btnLogout, btnAppUpdate, btnReportUpload;
ServiceViewModel serviceViewModel;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
EdgeToEdge.enable(this);
setContentView(R.layout.activity_dashboard);
setStatusBarColor();
ViewCompat.setOnApplyWindowInsetsListener(findViewById(R.id.main), (v, insets) -> {
Insets systemBars = insets.getInsets(WindowInsetsCompat.Type.systemBars());
v.setPadding(systemBars.left, systemBars.top, systemBars.right, systemBars.bottom);
return insets;
});
if (!Helper.isNetworkConnected(this)) {
Toast.makeText(this, "No Internet Connection", Toast.LENGTH_LONG).show();
}
initializeLayout();
btnCreateReport.setOnClickListener(v -> {
Intent intent = new Intent(this, QCTerryActivity.class);
startActivity(intent);
});
btnFetchProduct.setOnClickListener(v -> {
if (Helper.isNetworkConnected(this)) {
}
});
btnLogout.setOnClickListener(v -> {});
btnAppUpdate.setOnClickListener(v -> {});
btnReportUpload.setOnClickListener(v -> {});
}
public void initializeLayout() {
btnCreateReport = findViewById(R.id.btn_create_report);
btnFetchProduct = findViewById(R.id.btn_fetch_product);
btnLogout = findViewById(R.id.btn_logout);
btnAppUpdate = findViewById(R.id.btn_app_update);
btnReportUpload = findViewById(R.id.btn_reports_upload);
serviceViewModel = new ViewModelProvider(this).get(ServiceViewModel.class);
serviceViewModel.getLoadingState().observe(this, isLoading -> {
if (isLoading != null && isLoading) {
showProgressDialog();
} else {
dismissProgressDialog();
}
});
serviceViewModel.getErrorMessage().observe(this, errorResponse -> {
if (errorResponse.isEmpty()) {
Toast.makeText(this, "Something went wrong", Toast.LENGTH_SHORT).show();
} else {
Toast.makeText(this, errorResponse, Toast.LENGTH_SHORT).show();
}
});
serviceViewModel.getCheckpointsList().observe(this, checkpointsList -> {
if (!checkpointsList.isEmpty()) {
} else {
Toast.makeText(this, "List is empty", Toast.LENGTH_SHORT).show();
}
});
}
private void getScreenSize() {
DisplayMetrics metrics = new DisplayMetrics();
getWindowManager().getDefaultDisplay().getRealMetrics(metrics);
int widthPixels = metrics.widthPixels;
int heightPixels = metrics.heightPixels;
Log.e("Width: ",""+widthPixels);
Log.e("Height: ",""+heightPixels);
float widthInches = widthPixels / metrics.xdpi;
float heightInches = heightPixels / metrics.ydpi;
double diagonalInches = Math.sqrt(
Math.pow(widthInches, 2) +
Math.pow(heightInches, 2)
);
Log.e("Diagonal: ",""+diagonalInches);
int sw = getResources().getConfiguration().smallestScreenWidthDp;
Log.e("SWDP", "Smallest width dp = " + sw);
}
private void setStatusBarColor() {
Window window = getWindow();
window.addFlags(WindowManager.LayoutParams.FLAG_DRAWS_SYSTEM_BAR_BACKGROUNDS);
window.clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
window.setStatusBarColor(ContextCompat.getColor(this, R.color.theme_color));
WindowInsetsControllerCompat controller =
WindowCompat.getInsetsController(window, window.getDecorView());
controller.setAppearanceLightStatusBars(false);
}
public void setAppVersion() {
PackageManager packageManager = getPackageManager();
try {
PackageInfo info = packageManager.getPackageInfo( getPackageName(), 0 );
//txtVersion.setText(String.format("Version: %s", info.versionName));
} catch ( Exception e ) {
e.printStackTrace();
}
}
public void showProgressDialog() {
ProgressDialogFragment progressDialog = new ProgressDialogFragment();
progressDialog.setCancelable(false);
progressDialog.show(getSupportFragmentManager(), "progressDialog");
}
public void dismissProgressDialog() {
ProgressDialogFragment progressDialog = (ProgressDialogFragment) getSupportFragmentManager().findFragmentByTag("progressDialog");
if (progressDialog != null) {
progressDialog.dismiss();
}
}
}

View File

@ -0,0 +1,254 @@
package com.utopiaindustries.qc_android.activities;
import android.content.Intent;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.net.Uri;
import android.os.Bundle;
import android.widget.Button;
import android.widget.EditText;
import android.widget.ImageView;
import android.widget.TextView;
import android.widget.Toast;
import androidx.activity.EdgeToEdge;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.graphics.Insets;
import androidx.core.view.ViewCompat;
import androidx.core.view.WindowInsetsCompat;
import androidx.lifecycle.ViewModelProvider;
import com.utopiaindustries.qc_android.R;
import com.utopiaindustries.qc_android.apiservice.ApiService;
import com.utopiaindustries.qc_android.apiservice.ApiServiceFactory;
import com.utopiaindustries.qc_android.apiservice.GiteaClient;
import com.utopiaindustries.qc_android.apiservice.GiteaRestService;
import com.utopiaindustries.qc_android.helper.Helper;
import com.utopiaindustries.qc_android.helper.Preference;
import com.utopiaindustries.qc_android.helper.PropertyReader;
import com.utopiaindustries.qc_android.helper.Release;
import com.utopiaindustries.qc_android.utils.ProgressDialogFragment;
import com.utopiaindustries.qc_android.viewmodels.ServiceViewModel;
import java.util.Properties;
import retrofit2.Call;
import retrofit2.Callback;
import retrofit2.Response;
public class LoginActivity extends AppCompatActivity {
EditText tfEmail, tfPassword;
Button btnLogin;
ServiceViewModel serviceViewModel;
ApiService apiService;
TextView txtVersion;
ImageView imgCheckUpdate;
private String currentVersionName;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
EdgeToEdge.enable(this);
setContentView(R.layout.activity_login);
ViewCompat.setOnApplyWindowInsetsListener(findViewById(R.id.main), (v, insets) -> {
Insets systemBars = insets.getInsets(WindowInsetsCompat.Type.systemBars());
v.setPadding(systemBars.left, systemBars.top, systemBars.right, systemBars.bottom);
return insets;
});
/*ViewCompat.setOnApplyWindowInsetsListener(findViewById(R.id.main), (v, insets) -> {
Insets bars = insets.getInsets(WindowInsetsCompat.Type.systemBars() | WindowInsetsCompat.Type.ime());
v.setPadding(bars.left, bars.top, bars.right, bars.bottom);
return insets;
});*/
if (!Helper.isNetworkConnected(this)) {
Toast.makeText(this, "No Internet Connection", Toast.LENGTH_LONG).show();
}
initializeLayout();
btnLogin.setOnClickListener(v -> {
Preference.setMyBooleanPref(Helper.project_file, "isLoggedIn", getApplicationContext(), true);
Preference.setMyStringPref(Helper.project_file,Helper.logInUser,this,tfEmail.getText().toString());
Intent intent = new Intent(this, DashboardActivity.class);
startActivity(intent);
overridePendingTransition(R.anim.slide_in_right, R.anim.slide_out_left);
finish();
/*if (isValidate()) {
loginViewModel.isUserAuthenticated(tfEmail.getText().toString().trim(),
tfPassword.getText().toString(),
new String[]{"ROLE_UIM_HSE_APP_ACCESS_YES"});
*//*Preference.setMyBooleanPref(Helper.project_file, "isLoggedIn", getApplicationContext(), true);
Preference.setMyStringPref(Helper.project_file,Helper.logInUser,this,tfEmail.getText().toString());
Intent intent = new Intent(this, DashboardActivity.class);
startActivity(intent);
overridePendingTransition(R.anim.slide_in_right, R.anim.slide_out_left);
finish();*//*
}*/
});
imgCheckUpdate.setOnClickListener(v -> {
if (!Helper.isNetworkConnected(this)) {
Toast.makeText(this, "No Internet Connection", Toast.LENGTH_LONG).show();
}
else {
checkForUpdates();
}
});
}
public void setAppVersion() {
PackageManager packageManager = getPackageManager();
try {
PackageInfo info = packageManager.getPackageInfo( getPackageName(), 0 );
txtVersion.setText(String.format("Version: %s", info.versionName));
String currentVersion = String.format("v-%s", String.valueOf(info.versionName));
//StorageManager.getInstance().setAppVersion(currentVersion);
} catch ( Exception e ) {
e.printStackTrace();
}
}
public void initializeLayout() {
tfEmail = findViewById(R.id.tf_email);
tfPassword = findViewById(R.id.tf_password);
btnLogin = findViewById(R.id.btn_login);
txtVersion = findViewById(R.id.txt_version);
imgCheckUpdate = findViewById(R.id.img_check_update);
apiService = ApiServiceFactory.getApiService();
serviceViewModel = new ViewModelProvider(this).get(ServiceViewModel.class);
serviceViewModel.getLoadingState().observe(this, isLoading -> {
if (isLoading != null && isLoading) {
showProgressDialog();
} else {
dismissProgressDialog();
}
});
serviceViewModel.getErrorMessage().observe(this, errorResponse -> {
if (errorResponse.isEmpty()) {
Toast.makeText(this, "Something went wrong", Toast.LENGTH_SHORT).show();
} else {
Toast.makeText(this, errorResponse, Toast.LENGTH_SHORT).show();
}
});
serviceViewModel.getLoginUser().observe(this, loginUser -> {
if (loginUser) {
Preference.setMyBooleanPref(Helper.project_file, "isLoggedIn", getApplicationContext(), true);
Preference.setMyStringPref(Helper.project_file, Helper.logInUser, this, tfEmail.getText().toString());
Intent intent = new Intent(this, DashboardActivity.class);
startActivity(intent);
finish();
overridePendingTransition(R.anim.slide_in_right, R.anim.slide_out_left);
} else {
Toast.makeText(this, "Login Failed", Toast.LENGTH_SHORT).show();
}
});
setAppVersion();
}
public boolean isValidate() {
boolean returnValue = true;
String message = "";
if (tfPassword.getText().toString().isEmpty()) {
message = "Please enter password.";
returnValue = false;
}
/*if (!Helper.isValidEmail(tfEmail.getText().toString())) {
message = "Please enter valid email.";
returnValue = false;
}*/
if (tfEmail.getText().toString().isEmpty()) {
message = "Please enter user name.";
returnValue = false;
}
if (!returnValue) {
Toast.makeText(this, message, Toast.LENGTH_SHORT).show();
}
return returnValue;
}
public void showProgressDialog() {
ProgressDialogFragment progressDialog = new ProgressDialogFragment();
progressDialog.setCancelable(false);
progressDialog.show(getSupportFragmentManager(), "progressDialog");
}
public void dismissProgressDialog() {
ProgressDialogFragment progressDialog = (ProgressDialogFragment) getSupportFragmentManager().findFragmentByTag("progressDialog");
if (progressDialog != null) {
progressDialog.dismiss();
}
}
/**
* check for updates
*/
private void checkForUpdates() {
PackageManager packageManager = getPackageManager();
try {
PackageInfo info = packageManager.getPackageInfo( getPackageName(), 0 );
currentVersionName = info.versionName;
} catch ( Exception e ) {
e.printStackTrace();
}
// adjust views
showProgressDialog();
// get properties
Properties properties = PropertyReader.getProperties( LoginActivity.this );
String token = properties.getProperty( "gitea_access_token" );
// send request
GiteaRestService restService = GiteaClient.getInstance().create( GiteaRestService.class );
Call<Release> call = restService.getLatestRelease( token );
call.enqueue( new Callback<Release>() {
@Override
public void onResponse( Call<Release> call, Response<Release> response ) {
if ( response.body() != null ) {
Release release = response.body();
try {
float releaseVersion = Float.parseFloat( release.getName().replace( "v", "" ) );
float currentVersion = Float.parseFloat( currentVersionName.replace( "v", "" ) );
if ( releaseVersion > currentVersion && !Helper.isNullOrEmpty( release.getDownloadUrl() ) ) {
String msg = String.format( "Updating to version %s", release.getName() );
Toast.makeText( LoginActivity.this, msg, Toast.LENGTH_LONG ).show();
Intent intent = new Intent( Intent.ACTION_VIEW, Uri.parse( release.getDownloadUrl() ) );
startActivity( intent );
//startApkDownloadAndPerformUpdate( release.getDownloadUrl(), release.getName() );
} else {
Toast.makeText( LoginActivity.this, "Latest version is already installed", Toast.LENGTH_LONG ).show();
}
} catch ( Exception e ) {
Toast.makeText( LoginActivity.this, e.getMessage(), Toast.LENGTH_LONG ).show();
e.printStackTrace();
}
}
dismissProgressDialog();
}
@Override
public void onFailure(Call<Release> call, Throwable t ) {
dismissProgressDialog();
t.printStackTrace();
}
} );
}
}

View File

@ -0,0 +1,43 @@
package com.utopiaindustries.qc_android.activities;
import android.os.Bundle;
import androidx.activity.EdgeToEdge;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.graphics.Insets;
import androidx.core.view.ViewCompat;
import androidx.core.view.WindowInsetsCompat;
import androidx.navigation.NavController;
import androidx.navigation.Navigation;
import com.utopiaindustries.qc_android.R;
public class QCTerryActivity extends AppCompatActivity {
private NavController navController;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
EdgeToEdge.enable(this);
setContentView(R.layout.activity_qcterry);
ViewCompat.setOnApplyWindowInsetsListener(findViewById(R.id.main), (v, insets) -> {
Insets systemBars = insets.getInsets(WindowInsetsCompat.Type.systemBars());
v.setPadding(systemBars.left, systemBars.top, systemBars.right, systemBars.bottom);
return insets;
});
initializeLayout();
}
private void initializeLayout() {
// Setup Navigation
navController = Navigation.findNavController(this, R.id.nav_host_fragment);
}
@Override
public boolean onSupportNavigateUp() {
return navController.navigateUp() || super.onSupportNavigateUp();
}
}

View File

@ -0,0 +1,201 @@
package com.utopiaindustries.qc_android.adapters;
import android.content.Context;
import android.net.Uri;
import android.text.Editable;
import android.text.TextWatcher;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.CheckBox;
import android.widget.CompoundButton;
import android.widget.EditText;
import android.widget.ImageButton;
import android.widget.TextView;
import androidx.annotation.NonNull;
import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView;
import com.utopiaindustries.qc_android.R;
import com.utopiaindustries.qc_android.models.InspectionCheckPoint;
import java.util.List;
public class CheckPointAdapter extends RecyclerView.Adapter<CheckPointAdapter.CheckPointViewHolder> {
private Context context;
private List<InspectionCheckPoint> checkPointList;
private OnCheckPointClickListener listener;
ImageAdapter imageAdapter;
public interface OnCheckPointClickListener {
void onImagePickerClicked(int position);
void onImageClicked(int parentPosition, int imagePosition, Uri imageUri);
void onImageDeleteClicked(int parentPosition, int imagePosition);
void onDeleteAllImagesClicked(int position);
void onRemarksChanged(int position, String remarks);
void onCheckBoxChanged(int position, boolean isOkChecked, boolean isNoChecked);
}
public CheckPointAdapter(Context context, List<InspectionCheckPoint> checkPointList,
OnCheckPointClickListener listener) {
this.context = context;
this.checkPointList = checkPointList;
this.listener = listener;
}
@NonNull
@Override
public CheckPointViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
View view = LayoutInflater.from( parent.getContext() ).inflate(R.layout.check_point, parent, false);
return new CheckPointViewHolder(view);
}
@Override
public void onBindViewHolder(@NonNull CheckPointViewHolder holder, int position) {
InspectionCheckPoint item = checkPointList.get(position);
// Set title
holder.tvTitle.setText(item.getTitle());
// Set remarks
holder.etRemarks.setText(item.getRemarks());
// Set checkboxes (mutually exclusive)
holder.cbOk.setChecked(item.isOkChecked());
holder.cbNo.setChecked(item.isNoChecked());
// Setup nested RecyclerView for images
setupImagesRecyclerView(holder, position, item);
// Setup listeners
setupListeners(holder, position);
}
private void setupImagesRecyclerView(CheckPointViewHolder holder, int parentPosition,
InspectionCheckPoint item) {
// Create and setup images adapter
/*ImageAdapter imagesAdapter = new ImageAdapter(item.getImages(),
new ImageAdapter.OnImageClickListener() {
@Override
public void onImageClick(int imagePosition, Uri imageUri) {
if (listener != null) {
listener.onImageClicked(parentPosition, imagePosition, imageUri);
}
}
@Override
public void onImageDelete(int imagePosition) {
if (listener != null) {
listener.onImageDeleteClicked(parentPosition, imagePosition);
}
}
});*/
imageAdapter = new ImageAdapter(item.getImages(), context);
holder.imageRecyclerView.setLayoutManager(new LinearLayoutManager(context, LinearLayoutManager.HORIZONTAL, false));
holder.imageRecyclerView.setAdapter(imageAdapter);
}
private void setupListeners(CheckPointViewHolder holder, int position) {
// Remarks text change listener
holder.etRemarks.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
public void afterTextChanged(Editable s) {
if (listener != null) {
listener.onRemarksChanged(position, s.toString());
}
}
});
// OK checkbox listener
holder.cbOk.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
@Override
public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
if (isChecked) {
holder.cbNo.setChecked(false);
checkPointList.get(position).setOkChecked(true);
checkPointList.get(position).setNoChecked(false);
} else {
checkPointList.get(position).setOkChecked(false);
}
if (listener != null) {
listener.onCheckBoxChanged(position,
checkPointList.get(position).isOkChecked(),
checkPointList.get(position).isNoChecked());
}
}
});
// NO checkbox listener
holder.cbNo.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
@Override
public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
if (isChecked) {
holder.cbOk.setChecked(false);
checkPointList.get(position).setNoChecked(true);
checkPointList.get(position).setOkChecked(false);
} else {
checkPointList.get(position).setNoChecked(false);
}
if (listener != null) {
listener.onCheckBoxChanged(position,
checkPointList.get(position).isOkChecked(),
checkPointList.get(position).isNoChecked());
}
}
});
// Image picker button
holder.btnImagePicker.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if (listener != null) {
listener.onImagePickerClicked(position);
}
}
});
// Delete all images button
holder.btnDeleteAllImages.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
if (listener != null) {
listener.onDeleteAllImagesClicked(position);
}
}
});
}
@Override
public int getItemCount() {
return checkPointList.size();
}
public static class CheckPointViewHolder extends RecyclerView.ViewHolder {
TextView tvTitle;
CheckBox cbOk, cbNo;
EditText etRemarks;
ImageButton btnImagePicker, btnDeleteAllImages;
RecyclerView imageRecyclerView;
public CheckPointViewHolder(@NonNull View itemView) {
super(itemView);
tvTitle = itemView.findViewById(R.id.check_point_title);
cbOk = itemView.findViewById(R.id.check_point_ok);
cbNo = itemView.findViewById(R.id.check_point_no);
etRemarks = itemView.findViewById(R.id.check_point_remarks);
imageRecyclerView = itemView.findViewById(R.id.image_recycler_view);
btnImagePicker = itemView.findViewById(R.id.image_picker);
btnDeleteAllImages = itemView.findViewById(R.id.delete_image);
}
}
}

View File

@ -0,0 +1,84 @@
package com.utopiaindustries.qc_android.adapters;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
import androidx.recyclerview.widget.RecyclerView;
import com.bumptech.glide.Glide;
import com.bumptech.glide.request.RequestOptions;
import com.utopiaindustries.qc_android.R;
import com.utopiaindustries.qc_android.utils.FullScreenImageDialog;
import java.util.List;
public class ImageAdapter extends RecyclerView.Adapter<ImageAdapter.ImageViewHolder> {
//private byte[] file;
//private List<String> imageList;
List<byte[]> imageList;
private Context context;
public ImageAdapter(List<byte[]> imageList, Context context) {
this.imageList = imageList;
this.context = context;
}
@NonNull
@Override
public ImageViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.item_image, parent, false);
return new ImageViewHolder(view);
}
@Override
public void onBindViewHolder(@NonNull ImageViewHolder holder, int position) {
//holder.imageView.setImageBitmap(null);
holder.imageView.setImageResource(R.drawable.img_load);
//Bitmap bitmap = BitmapFactory.decodeByteArray( imageList.get(position), 0 , imageList.get(position).length );
//holder.imageView.setImageBitmap( bitmap );
/*Glide.with(context)
.load(bitmap) // Load the Bitmap directly
.placeholder(R.drawable.img_load)
.apply(new RequestOptions().centerCrop()) // Optional: apply transformations like center crop
.into(holder.imageView);*/
Glide.with(context)
.load(imageList.get(position)) // Glide will handle the decoding and placeholder
.placeholder(R.drawable.img_load)
.apply(new RequestOptions().centerCrop())
.into(holder.imageView);
Bitmap bitmap = BitmapFactory.decodeByteArray(imageList.get(position), 0, imageList.get(position).length);
holder.imageView.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
FullScreenImageDialog dialog = new FullScreenImageDialog(bitmap);
dialog.show(((AppCompatActivity) v.getContext()).getSupportFragmentManager(), "FullScreenDialog");
}
});
}
@Override
public int getItemCount() {
return (imageList != null) ? imageList.size() : 0;
}
public class ImageViewHolder extends RecyclerView.ViewHolder {
ImageView imageView;
public ImageViewHolder(@NonNull View itemView) {
super(itemView);
imageView = itemView.findViewById(R.id.imageView);
}
}
}

View File

@ -0,0 +1,62 @@
package com.utopiaindustries.qc_android.adapters;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.widget.TextView;
import androidx.annotation.NonNull;
import androidx.fragment.app.Fragment;
import androidx.fragment.app.FragmentManager;
import androidx.fragment.app.FragmentStatePagerAdapter;
import com.utopiaindustries.qc_android.R;
import com.utopiaindustries.qc_android.fragments.DraftFragment;
import com.utopiaindustries.qc_android.fragments.ReportsFragment;
@SuppressWarnings("deprecation")
public class PagerAdapter extends FragmentStatePagerAdapter {
String tabTitles[] = new String[]{"Submitted Reports", "Editable / Draft Reports"};
Context context;
public PagerAdapter(FragmentManager fm, Context context) {
super(fm, BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT);
this.context = context;
}
@Override
public int getCount() {
return tabTitles.length;
}
@NonNull
@Override
public Fragment getItem(int position) {
switch (position) {
case 0:
return new ReportsFragment();
case 1:
return new DraftFragment();
}
return null;
}
@Override
public CharSequence getPageTitle(int position) {
// Generate title based on item position
// return tabTitles[position];
return null;
}
public View getTabView(int position) {
View tab = LayoutInflater.from(context).inflate(R.layout.custom_tab, null);
TextView tv = (TextView) tab.findViewById(R.id.txt_name);
tv.setText(tabTitles[position]);
// tv.setTextSize(TypedValue.COMPLEX_UNIT_SP, context.getResources().getDimension(R.dimen.tab_text_size));
return tab;
}
}

View File

@ -0,0 +1,53 @@
package com.utopiaindustries.qc_android.apiservice;
import com.utopiaindustries.qc_android.models.InspectionCheckPoint;
import java.util.List;
import retrofit2.Call;
import retrofit2.http.Body;
import retrofit2.http.GET;
import retrofit2.http.POST;
import retrofit2.http.Query;
public interface ApiService {
/*@GET("rest/uic/hse/get-hse-report-data")
Call<HseResponse> getHseData();*/
/*@GET("rest/uic/hse/getHseReports")
Call<ReportListResponse> getHseReportList(
@Query("username") String username
);*/
/*@GET("rest/uic/hse/get-hse-employee-by-id")
Call<EmployeeInfoResponse> getEmployeeData(
@Query("employee-id") String empId
);*/
/*@GET("rest/uic/hse/get-daily-wages-contractor-data-by-visit-id")
Call<DailyWageResponse> getDailyWageData(
@Query("visit-id") String visitorId
);*/
/*@POST("rest/uic/hse/save-hse-report")
Call<HseSaveResponse> saveHseReport(
@Body HseReportRequest request
);*/
@POST("rest/authentication/authenticate-user")
Call<Boolean> isUserAuthenticated(
@Query("username") String username,
@Query("password") String password,
@Query("roles") String[] roles
);
/*@POST("rest/uic/hse/save-draft-count")
Call<HseDraftResponse> sendDraftCount(
@Query("username") String username,
@Query("count") String password
);*/
@GET("rest/uic/inspection-report/checkpoints")
Call<List<InspectionCheckPoint>> fetchCheckPoints();
}

View File

@ -0,0 +1,17 @@
package com.utopiaindustries.qc_android.apiservice;
import retrofit2.Retrofit;
public class ApiServiceFactory {
private static ApiService apiService;
public synchronized static ApiService getApiService() {
if ( apiService == null ) {
Retrofit retrofit = RetrofitClient.getClient();
apiService = retrofit.create( ApiService.class );
}
return apiService;
}
}

View File

@ -0,0 +1,27 @@
package com.utopiaindustries.qc_android.apiservice;
import retrofit2.Retrofit;
import retrofit2.converter.jackson.JacksonConverterFactory;
public class GiteaClient {
private static Retrofit retrofit;
private static final String BASE_URL = "https://git.utopiadeals.com/api/v1/repos/UIND/HSE-Android-Releases/";
/**
* get retrofit instance
*/
public static Retrofit getInstance() {
if ( retrofit == null ) {
try {
retrofit = new Retrofit.Builder()
.baseUrl( BASE_URL )
.addConverterFactory( JacksonConverterFactory.create() )
.client( SSLCheckHttpClient.getOkHttpClient() )
.build();
} catch (Exception e) {
throw new RuntimeException(e);
}
}
return retrofit;
}
}

View File

@ -0,0 +1,16 @@
package com.utopiaindustries.qc_android.apiservice;
import com.utopiaindustries.qc_android.helper.Release;
import retrofit2.Call;
import retrofit2.http.GET;
import retrofit2.http.Query;
public interface GiteaRestService {
/**
* get latest release
*/
@GET( "releases/latest" )
Call<Release> getLatestRelease(@Query( "access_token" ) String accessToken );
}

View File

@ -0,0 +1,49 @@
package com.utopiaindustries.qc_android.apiservice;
import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
import java.util.concurrent.TimeUnit;
import okhttp3.OkHttpClient;
import retrofit2.Retrofit;
import retrofit2.converter.gson.GsonConverterFactory;
public class RetrofitClient {
//Live url
//private final static String BASE_URL = "https://portal.utopiaindustries.pk/uind/";
//Test Url
private final static String BASE_URL = "http://192.168.90.22:8081/uind/";
private static Retrofit retrofit;
public synchronized static Retrofit getClient() {
Gson gson = new GsonBuilder()
//.registerTypeAdapter(Boolean.class, new BooleanTypeAdapter())
.create();
if (retrofit == null) {
try {
OkHttpClient okHttpClient = new OkHttpClient.Builder()
.connectTimeout(60, TimeUnit.SECONDS)
.readTimeout(60, TimeUnit.SECONDS)
.writeTimeout(60, TimeUnit.SECONDS)
.build();
retrofit = new Retrofit.Builder()
.baseUrl(BASE_URL)
.client(okHttpClient)
.addConverterFactory(GsonConverterFactory.create(gson))
.build();
} catch (Exception e) {
throw new RuntimeException(e);
}
}
return retrofit;
}
}

View File

@ -0,0 +1,52 @@
package com.utopiaindustries.qc_android.apiservice;
import java.security.KeyManagementException;
import java.security.NoSuchAlgorithmException;
import java.security.cert.CertificateException;
import java.security.cert.X509Certificate;
import java.util.concurrent.TimeUnit;
import javax.net.ssl.SSLContext;
import javax.net.ssl.SSLSocketFactory;
import javax.net.ssl.TrustManager;
import javax.net.ssl.X509TrustManager;
import okhttp3.OkHttpClient;
public class SSLCheckHttpClient {
public static OkHttpClient getOkHttpClient() throws NoSuchAlgorithmException, KeyManagementException {
// Create a trust manager that does not validate certificate chains
final TrustManager[] trustAllCerts = new TrustManager[]{
new X509TrustManager() {
@Override
public void checkClientTrusted(X509Certificate[] chain, String authType) throws CertificateException {
}
@Override
public void checkServerTrusted(X509Certificate[] chain, String authType) throws CertificateException {
}
@Override
public X509Certificate[] getAcceptedIssuers() {
return new X509Certificate[0];
}
}
};
// Install the all-trusting trust manager
final SSLContext sslContext = SSLContext.getInstance("SSL");
sslContext.init(null, trustAllCerts, new java.security.SecureRandom());
// Create an ssl socket factory with our all-trusting manager
final SSLSocketFactory sslSocketFactory = sslContext.getSocketFactory();
return new OkHttpClient.Builder()
.sslSocketFactory(sslSocketFactory, (X509TrustManager) trustAllCerts[0])
.hostnameVerifier((hostname, session) -> true)
.connectTimeout(60, TimeUnit.SECONDS)
.readTimeout(60, TimeUnit.SECONDS)
.writeTimeout(60, TimeUnit.SECONDS)
.build();
}
}

View File

@ -0,0 +1,83 @@
package com.utopiaindustries.qc_android.db;
import android.annotation.SuppressLint;
import android.content.ContentValues;
import android.content.Context;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import com.utopiaindustries.qc_android.models.InspectionCheckPoint;
import com.utopiaindustries.qc_android.utils.DatabaseExecutor;
import java.util.ArrayList;
import java.util.List;
public class CheckpointRepository {
private final DatabaseHelper dbHelper;
private final SQLiteDatabase database;
//private final ExecutorService executorService;
public CheckpointRepository(Context context ) {
dbHelper = new DatabaseHelper ( context );
database = dbHelper.getWritableDatabase();
//executorService = Executors.newSingleThreadExecutor();
}
/*
* Insert list of checkpoints in batch
* */
public void insert( List<InspectionCheckPoint> checkPoints ) {
DatabaseExecutor.getExecutor().execute(() -> {
database.beginTransaction();
try {
for ( InspectionCheckPoint checkPoint : checkPoints ) {
ContentValues values = new ContentValues();
values.put(DatabaseHelper.CHECKPOINT_COLUMN_ID, checkPoint.getId() );
values.put(DatabaseHelper.CHECKPOINT_COLUMN_TITLE, checkPoint.getTitle() );
values.put(DatabaseHelper.CHECKPOINT_COLUMN_CATEGORY, checkPoint.getCategory() );
database.insertWithOnConflict(DatabaseHelper.CHECKPOINT_TABLE_NAME, null, values, SQLiteDatabase.CONFLICT_REPLACE);
}
database.setTransactionSuccessful();
} finally {
database.endTransaction();
close();
}
});
}
@SuppressLint("Range")
public List<InspectionCheckPoint> findAll( ) {
Cursor cursor = database.query( DatabaseHelper.CHECKPOINT_TABLE_NAME,
null,
null,
null,
null,
null,
null);
List<InspectionCheckPoint> checkPoints = new ArrayList<>();
if (cursor != null && cursor.moveToFirst()) {
do {
InspectionCheckPoint checkPoint = new InspectionCheckPoint();
checkPoint.setId(cursor.getLong( cursor.getColumnIndex(DatabaseHelper.CHECKPOINT_COLUMN_ID )));
checkPoint.setTitle(cursor.getString(cursor.getColumnIndex(DatabaseHelper.CHECKPOINT_COLUMN_TITLE) ));
checkPoint.setCategory(cursor.getString(cursor.getColumnIndex(DatabaseHelper.CHECKPOINT_COLUMN_CATEGORY )));
checkPoints.add( checkPoint );
} while ( cursor.moveToNext() );
cursor.close();
}
close();
return checkPoints;
}
/*
* Close the database
*/
public void close() {
dbHelper.close();
//executorService.shutdown();
}
}

View File

@ -0,0 +1,213 @@
package com.utopiaindustries.qc_android.db;
import android.content.Context;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteOpenHelper;
public class DatabaseHelper extends SQLiteOpenHelper {
private static final int DATABASE_VERSION = 1;
private static final String DATABASE_NAME = "inventory";
// Table and columns
public static final String PRODUCT_TABLE_NAME = "product";
public static final String PRODUCT_COLUMN_ASIN = "asin";
public static final String PRODUCT_COLUMN_PARENT_ASIN = "parentAsin";
public static final String PRODUCT_COLUMN_MARKETPLACE = "marketplace";
public static final String PRODUCT_COLUMN_SKU = "sku";
public static final String PRODUCT_COLUMN_TITLE = "title";
public static final String PRODUCT_COLUMN_HS_CODE = "hsCode";
public static final String PRODUCT_COLUMN_MODEL_NUMBER = "modelNumber";
public static final String PRODUCT_COLUMN_ITEM_PER_BOX = "itemPerBox";
public static final String PRODUCT_COLUMN_SM_COLOR = "smColor";
public static final String PRODUCT_COLUMN_SM_SIZE = "smSize";
public static final String PRODUCT_COLUMN_SM_ITEM_NAME = "smItemName";
public static final String PRODUCT_COLUMN_ITEM_PER_PACK = "itemPerPack";
public static final String PRODUCT_COLUMN_INVENTORY = "inventory";
public static final String PRODUCT_COLUMN_CATEGORY = "category";
public static final String PRODUCT_COLUMN_FNSKU = "fnsku";
//Table and Column for Inspection Label
public static final String INSPECTION_LABEL_TABLE_NAME = "inspection_label";
public static final String INSPECTION_LABEL_COLUMN_ID = "id";
public static final String INSPECTION_LABEL_COLUMN_MIN_LOT_SIZE = "minLotSize";
public static final String INSPECTION_LABEL_COLUMN_MAX_LOT_SIZE = "maxLotSize";
public static final String INSPECTION_LABEL_COLUMN_INSPECTION_LEVEL = "inspectionLevel";
public static final String INSPECTION_LABEL_COLUMN_SAMPLE_CODE = "sampleCode";
//Table and Column for Quality Label
public static final String QUALITY_LABEL_TABLE_NAME = "quality_label";
public static final String QUALITY_LABEL_COLUMN_ID = "id";
public static final String QUALITY_LABEL_COLUMN_SAMPLE_CODE = "sampleCode";
public static final String QUALITY_LABEL_COLUMN_SAMPLE_SIZE = "sampleSize";
public static final String QUALITY_LABEL_COLUMN_QUALITY_LEVEL = "qualityLevel";
public static final String QUALITY_LABEL_COLUMN_NO_ACCEPTED_DEFECTS = "noOfAcceptedDefects";
/*
* dimension table
*/
public static final String INSPECTION_DIMENSION_TABLE_NAME = "dimension";
public static final String DIMENSION_COLUMN_ID = "id";
public static final String DIMENSION_COLUMN_TITLE = "title";
public static final String DIMENSION_COLUMN_CATEGORY = "category";
/*
* checkpoints table
* */
public static final String CHECKPOINT_TABLE_NAME = "checkpoint";
public static final String CHECKPOINT_COLUMN_ID = "id";
public static final String CHECKPOINT_COLUMN_TITLE = "title";
public static final String CHECKPOINT_COLUMN_CATEGORY = "category";
/*
* defect table
* */
public static final String DEFECT_TABLE_NAME = "defect";
public static final String DEFECT_COLUMN_DEFECT = "defect";
public static final String DEFECT_COLUMN_CATEGORY = "category";
/*
* unit table
* */
public static final String UNIT_TABLE_NAME = "unit";
public static final String UNIT_COLUMN_ID = "id";
public static final String UNIT_COLUMN_TITLE = "title";
/*
* report table
* */
public static final String REPORT_TABLE_NAME = "report";
public static final String REPORT_COLUMN_ID = "id";
public static final String REPORT_COLUMN_CONTENT = "content";
public static final String REPORT_COLUMN_SYNCED = "synced";
/*
* draft report table
* */
public static final String DRAFT_REPORT_TABLE_NAME = "draft_report";
public static final String DRAFT_REPORT_COLUMN_ID = "id";
public static final String DRAFT_REPORT_COLUMN_CONTENT = "content";
//public static final String DRAFT_REPORT_COLUMN_FNSKU = "fnsku";
public static final String DRAFT_REPORT_COLUMN_SYNCED = "synced";
/*
* sku checkpoints standard
* */
public static final String CHECKPOINT_SKU_TABLE_NAME = "checkpoint_sku";
public static final String CHECKPOINT_SKU_SKU = "sku";
public static final String CHECKPOINT_SKU_MARKETPLACE = "marketplace";
public static final String CHECKPOINT_SKU_CHECKPOINT_ID = "checkpoint_id";
public static final String CHECKPOINT_SKU_STANDARD = "standard";
public DatabaseHelper(Context context) {
super(context, DATABASE_NAME, null, DATABASE_VERSION);
}
@Override
public void onCreate(SQLiteDatabase db) {
// creating product table
String CREATE_PRODUCT_TABLE = "CREATE TABLE " + PRODUCT_TABLE_NAME + "("
+ PRODUCT_COLUMN_ASIN + " TEXT,"
+ PRODUCT_COLUMN_PARENT_ASIN + " TEXT,"
+ PRODUCT_COLUMN_MARKETPLACE + " TEXT,"
+ PRODUCT_COLUMN_SKU + " TEXT,"
+ PRODUCT_COLUMN_TITLE + " TEXT,"
+ PRODUCT_COLUMN_HS_CODE + " TEXT,"
+ PRODUCT_COLUMN_MODEL_NUMBER + " TEXT,"
+ PRODUCT_COLUMN_ITEM_PER_BOX + " INTEGER,"
+ PRODUCT_COLUMN_SM_COLOR + " TEXT,"
+ PRODUCT_COLUMN_SM_SIZE + " TEXT,"
+ PRODUCT_COLUMN_SM_ITEM_NAME + " TEXT,"
+ PRODUCT_COLUMN_ITEM_PER_PACK + " INTEGER,"
+ PRODUCT_COLUMN_INVENTORY + " INTEGER,"
+ PRODUCT_COLUMN_CATEGORY + " TEXT,"
+ PRODUCT_COLUMN_FNSKU + " TEXT,"
+ "PRIMARY KEY (" + PRODUCT_COLUMN_ASIN + ", " + PRODUCT_COLUMN_MARKETPLACE + ", " + PRODUCT_COLUMN_SKU + ")"
+ ")";
String CREATE_INSPECTION_DIMENSION_TABLE = "CREATE TABLE " + INSPECTION_DIMENSION_TABLE_NAME + " (" +
DIMENSION_COLUMN_ID + " INTEGER PRIMARY KEY," +
DIMENSION_COLUMN_TITLE + " TEXT NOT NULL," +
DIMENSION_COLUMN_CATEGORY + " TEXT NOT NULL);";
String CREATE_CHECKPOINT_TABLE = "CREATE TABLE " + CHECKPOINT_TABLE_NAME + " (" +
CHECKPOINT_COLUMN_ID + " INTEGER PRIMARY KEY," +
CHECKPOINT_COLUMN_TITLE + " TEXT NOT NULL," +
CHECKPOINT_COLUMN_CATEGORY + " TEXT NOT NULL);";
String CREATE_DEFECT_TABLE = "CREATE TABLE " + DEFECT_TABLE_NAME + " (" +
DEFECT_COLUMN_DEFECT + " TEXT NOT NULL," +
DEFECT_COLUMN_CATEGORY + " TEXT NOT NULL," +
"PRIMARY KEY (" + DEFECT_COLUMN_DEFECT + ", " + DEFECT_COLUMN_CATEGORY + ")"
+ ")";
String CREATE_UNIT_TABLE = "CREATE TABLE " + UNIT_TABLE_NAME + " (" +
UNIT_COLUMN_ID + " INTEGER PRIMARY KEY," +
UNIT_COLUMN_TITLE + " TEXT NOT NULL )";
String CREATE_REPORT_TABLE = "CREATE TABLE " + REPORT_TABLE_NAME + " (" +
REPORT_COLUMN_ID + " INTEGER PRIMARY KEY AUTOINCREMENT," +
REPORT_COLUMN_CONTENT + " TEXT NOT NULL," +
REPORT_COLUMN_SYNCED + " INTEGER DEFAULT 0 CHECK(synced IN (0, 1))"
+ ")";
String CREATE_DRAFT_REPORT_TABLE = "CREATE TABLE " + DRAFT_REPORT_TABLE_NAME + " (" +
DRAFT_REPORT_COLUMN_ID + " INTEGER PRIMARY KEY AUTOINCREMENT, " +
DRAFT_REPORT_COLUMN_CONTENT + " TEXT NOT NULL, " +
// DRAFT_REPORT_COLUMN_FNSKU + " TEXT NOT NULL," +
DRAFT_REPORT_COLUMN_SYNCED + " INTEGER DEFAULT 0 CHECK(synced IN (0, 1))"
+ ")";
String CREATE_SKU_CHECKPOINT_TABLE = "CREATE TABLE " + CHECKPOINT_SKU_TABLE_NAME + " (" +
CHECKPOINT_SKU_SKU + " TEXT NOT NULL," +
CHECKPOINT_SKU_MARKETPLACE + " TEXT NOT NULL," +
CHECKPOINT_SKU_CHECKPOINT_ID + " INTEGER NOT NULL," +
CHECKPOINT_SKU_STANDARD + " TEXT," +
"PRIMARY KEY (" + CHECKPOINT_SKU_SKU + ", " + CHECKPOINT_SKU_MARKETPLACE + ", " + CHECKPOINT_SKU_CHECKPOINT_ID + "))";
String CREATE_INSPECTION_LABEL_TABLE = "CREATE TABLE " + INSPECTION_LABEL_TABLE_NAME + " (" +
INSPECTION_LABEL_COLUMN_ID + " INTEGER NOT NULL," +
INSPECTION_LABEL_COLUMN_MIN_LOT_SIZE + " INTEGER NOT NULL," +
INSPECTION_LABEL_COLUMN_MAX_LOT_SIZE + " INTEGER NOT NULL," +
INSPECTION_LABEL_COLUMN_INSPECTION_LEVEL + " INTEGER NOT NULL," +
INSPECTION_LABEL_COLUMN_SAMPLE_CODE + " TEXT NOT NULL )";
String CREATE_QUALITY_LABEL_TABLE = "CREATE TABLE " + QUALITY_LABEL_TABLE_NAME + " (" +
QUALITY_LABEL_COLUMN_ID + " INTEGER NOT NULL," +
QUALITY_LABEL_COLUMN_SAMPLE_CODE + " TEXT NOT NULL," +
QUALITY_LABEL_COLUMN_SAMPLE_SIZE + " INTEGER NOT NULL," +
QUALITY_LABEL_COLUMN_QUALITY_LEVEL + " REAL NOT NULL," +
QUALITY_LABEL_COLUMN_NO_ACCEPTED_DEFECTS + " INTEGER NOT NULL )";
db.execSQL( CREATE_PRODUCT_TABLE );
db.execSQL( CREATE_INSPECTION_DIMENSION_TABLE );
db.execSQL( CREATE_CHECKPOINT_TABLE );
db.execSQL( CREATE_DEFECT_TABLE );
db.execSQL( CREATE_UNIT_TABLE );
db.execSQL( CREATE_REPORT_TABLE );
db.execSQL( CREATE_DRAFT_REPORT_TABLE);
db.execSQL( CREATE_SKU_CHECKPOINT_TABLE );
db.execSQL( CREATE_INSPECTION_LABEL_TABLE );
db.execSQL( CREATE_QUALITY_LABEL_TABLE );
}
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
db.execSQL( "DROP TABLE IF EXISTS " + PRODUCT_TABLE_NAME);
db.execSQL( "DROP TABLE IF EXISTS " + INSPECTION_DIMENSION_TABLE_NAME );
db.execSQL( "DROP TABLE IF EXISTS " + CHECKPOINT_TABLE_NAME );
db.execSQL( "DROP TABLE IF EXISTS " + DEFECT_TABLE_NAME );
db.execSQL( "DROP TABLE IF EXISTS " + UNIT_TABLE_NAME );
db.execSQL( "DROP TABLE IF EXISTS " + REPORT_TABLE_NAME );
db.execSQL( "DROP TABLE IF EXISTS " + DRAFT_REPORT_TABLE_NAME );
db.execSQL( "DROP TABLE IF EXISTS " + CHECKPOINT_SKU_TABLE_NAME );
db.execSQL( "DROP TABLE IF EXISTS " + INSPECTION_LABEL_TABLE_NAME );
db.execSQL( "DROP TABLE IF EXISTS " + QUALITY_LABEL_TABLE_NAME );
onCreate(db);
}
}

View File

@ -0,0 +1,195 @@
package com.utopiaindustries.qc_android.fragments;
import android.content.Context;
import android.content.Intent;
import android.content.SharedPreferences;
import android.os.Bundle;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;
import androidx.fragment.app.Fragment;
import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView;
import com.utopiaindustries.qc_android.R;
import com.utopiaindustries.qc_android.utils.SelectListener;
import java.io.ByteArrayInputStream;
import java.io.ObjectInputStream;
import java.util.ArrayList;
public class DraftFragment extends Fragment implements SelectListener {
RecyclerView recyclerView;
TextView emptyReportTextView;
//private Store store;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
// Inflate the layout for this fragment
View view = inflater.inflate(R.layout.fragment_draft , container, false );
initializeViews( view );
return view;
}
private void initializeViews( View view ) {
recyclerView = view.findViewById( R.id.draft_recyclerview );
emptyReportTextView = view.findViewById(R.id.empty_report_text);
recyclerView.setLayoutManager( new LinearLayoutManager( getContext() ) );
//store = Store.getInstance();
//refreshReportData();
}
@Override
public void onResume() {
super.onResume();
//refreshReportData();
}
/*public void refreshReportData() {
try {
ReportRepository repository = new ReportRepository(getContext());
List<InspectionReportWrapper> reportWrappers = repository.findAllDraftReports();
List<InspectionReport> reports = getReports(reportWrappers);
// --- Filter expired reports ---
Iterator<InspectionReportWrapper> itWrapper = reportWrappers.iterator();
Iterator<InspectionReport> itReport = reports.iterator();
while (itWrapper.hasNext() && itReport.hasNext()) {
InspectionReportWrapper wrapper = itWrapper.next();
InspectionReport report = itReport.next();
if (report.getGeneratedAt() != null) {
if (DateTimeUtils.isOutside24Hours(report.getGeneratedAt())) {
Log.e("Outside-24-hours", "*****");
// Delete from DB
repository.deleteDraftRecordById(wrapper.getId());
// Remove from both lists
itWrapper.remove();
itReport.remove();
} else {
Log.e("Inside-24-hours", "*****");
}
}
}
DraftReportAdapter adapter = new DraftReportAdapter(reportWrappers, this, getContext(), reports);
recyclerView.setAdapter(adapter);
if (reportWrappers.isEmpty()) {
emptyReportTextView.setVisibility(View.VISIBLE);
} else {
emptyReportTextView.setVisibility(View.INVISIBLE);
}
} catch (Exception e) {
Log.e("Exception: ", "change");
}
}*/
/*public void refreshReportDataWithSaturdayException() {
try {
ReportRepository repository = new ReportRepository(getContext());
List<InspectionReportWrapper> reportWrappers = repository.findAllDraftReports();
List<InspectionReport> reports = getReports(reportWrappers);
Iterator<InspectionReportWrapper> itWrapper = reportWrappers.iterator();
Iterator<InspectionReport> itReport = reports.iterator();
while (itWrapper.hasNext() && itReport.hasNext()) {
InspectionReportWrapper wrapper = itWrapper.next();
InspectionReport report = itReport.next();
String generatedAtString = report.getGeneratedAt();
if (generatedAtString != null && !generatedAtString.isEmpty()) {
// Check if report should be deleted
boolean shouldDelete = DateTimeUtils.isReportExpired(generatedAtString);
if (shouldDelete) {
Log.e("DELETE", "Deleting report: " + generatedAtString);
// Delete from database
repository.deleteDraftRecordById(wrapper.getId());
// Remove from lists
itWrapper.remove();
itReport.remove();
} else {
Log.e("KEEP", "Keeping report: " + generatedAtString);
}
}
}
// Update UI
DraftReportAdapter adapter = new DraftReportAdapter(reportWrappers, this, getContext(), reports);
recyclerView.setAdapter(adapter);
emptyReportTextView.setVisibility(reportWrappers.isEmpty() ? View.VISIBLE : View.INVISIBLE);
} catch (Exception e) {
Log.e("refreshReportData", "Error: ", e);
}
}*/
/*@Override
public void onItemClicked(InspectionReportWrapper reportWrapper) {
//Toast.makeText(getContext(), String.valueOf(reportWrapper.getId()), Toast.LENGTH_SHORT).show();
SharedPreferences sharedPreferences = getContext().getSharedPreferences("login_prefs", Context.MODE_PRIVATE);
SharedPreferences.Editor editor = sharedPreferences.edit();
editor.putString("draftReportId", String.valueOf(reportWrapper.getId()));
editor.apply();
Intent intent = new Intent( getContext(), MainActivity.class );
intent.putExtra("fromClass", "Draft");
intent.putExtra("draftID", String.valueOf(reportWrapper.getId()));
startActivity( intent );
}*/
/*private List<InspectionReport> getReports(List<InspectionReportWrapper> wrappers ) {
List<InspectionReport> reports = new ArrayList<>();
if( ! wrappers.isEmpty() ){
for( InspectionReportWrapper wrapper : wrappers ){
try {
// get file from path
byte[] result = FileUtils.readFile( wrapper.getContent() );
*//*FileUtils.copyFile(this, wrapper.getContent(), "Reports");
Object obj = FileUtils.readSerializedObject(wrapper.getContent());
if (obj != null) {
FileUtils.writeObjectToReadableFile(this, obj, "report_readable.txt");
}*//*
ByteArrayInputStream byteIn = new ByteArrayInputStream( result );
ObjectInputStream in = new ObjectInputStream(byteIn);
InspectionReport report = ( InspectionReport ) in.readObject();
report.setWrapperId( wrapper.getId() );
report.setFilePath( wrapper.getContent() );
reports.add( report );
} catch (Exception | OutOfMemoryError e){
e.printStackTrace();
}
}
}
return reports;
}*/
}

View File

@ -0,0 +1,293 @@
package com.utopiaindustries.qc_android.fragments;
import android.Manifest;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.net.Uri;
import android.os.Build;
import android.os.Bundle;
import android.provider.MediaStore;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Button;
import android.widget.EditText;
import android.widget.ImageView;
import android.widget.Toast;
import androidx.activity.result.ActivityResultCallback;
import androidx.activity.result.ActivityResultLauncher;
import androidx.activity.result.contract.ActivityResultContracts;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.core.content.ContextCompat;
import androidx.core.content.FileProvider;
import androidx.fragment.app.Fragment;
import androidx.lifecycle.ViewModelProvider;
import androidx.navigation.Navigation;
import com.utopiaindustries.qc_android.R;
import com.utopiaindustries.qc_android.viewmodels.DataEntryViewModel;
import java.io.File;
import java.io.IOException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Locale;
import java.util.Map;
public class FragmentQcTerryOne extends Fragment {
private EditText edtFirstName, edtLastName, edtEmail;
private Button btnNext;
private DataEntryViewModel viewModel;
private ActivityResultLauncher<String[]> requestMultiplePermissionsLauncher;
// Activity Result Launchers
private ActivityResultLauncher<Intent> cameraLauncher;
private ActivityResultLauncher<String> galleryLauncher;
private ImageView imageView;
private Button btnCamera, btnGallery;
private Uri photoUri;
private static final int PENDING_CAMERA = 1;
private static final int PENDING_GALLERY = 2;
private int pendingAction = 0;
@Nullable
@Override
public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container,
@Nullable Bundle savedInstanceState) {
// Inflate the layout
View view = inflater.inflate(R.layout.fragment_qc_terry_one, container, false);
// Initialize views
edtFirstName = view.findViewById(R.id.edtFirstName);
edtLastName = view.findViewById(R.id.edtLastName);
edtEmail = view.findViewById(R.id.edtEmail);
btnNext = view.findViewById(R.id.btnNext);
imageView = view.findViewById(R.id.imageView);
btnCamera = view.findViewById(R.id.btnCamera);
btnGallery = view.findViewById(R.id.btnGallery);
return view;
}
@Override
public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) {
super.onViewCreated(view, savedInstanceState);
// Initialize ViewModel
viewModel = new ViewModelProvider(requireActivity()).get(DataEntryViewModel.class);
// Load existing data if any
if (viewModel.getFirstName().getValue() != null) {
edtFirstName.setText(viewModel.getFirstName().getValue());
}
if (viewModel.getLastName().getValue() != null) {
edtLastName.setText(viewModel.getLastName().getValue());
}
if (viewModel.getEmail().getValue() != null) {
edtEmail.setText(viewModel.getEmail().getValue());
}
if (viewModel.getSelectedImageUri().getValue() != null) {
photoUri = viewModel.getSelectedImageUri().getValue();
imageView.setImageURI(photoUri);
}
// Setup button click
btnNext.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// Save data to ViewModel
viewModel.setFirstName(edtFirstName.getText().toString());
viewModel.setLastName(edtLastName.getText().toString());
viewModel.setEmail(edtEmail.getText().toString());
// Navigate to next screen
Navigation.findNavController(v)
.navigate(R.id.action_fragmentOne_to_fragmentTwo);
}
});
// Initialize Activity Result Launchers
initializeLaunchers();
btnCamera.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
checkPermissionAndOpenCamera();
}
});
btnGallery.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
checkPermissionAndOpenGallery();
}
});
}
private void initializeLaunchers() {
requestMultiplePermissionsLauncher = registerForActivityResult(
new ActivityResultContracts.RequestMultiplePermissions(),
new ActivityResultCallback<Map<String, Boolean>>() {
@Override
public void onActivityResult(Map<String, Boolean> result) {
boolean allGranted = true;
for (Boolean granted : result.values()) {
if (!granted) {
allGranted = false;
break;
}
}
if (allGranted) {
// All permissions granted, proceed based on what was requested
if (pendingAction == PENDING_CAMERA) {
openCamera();
} else if (pendingAction == PENDING_GALLERY) {
openGallery();
}
} else {
// Some permissions denied
Toast.makeText(getContext(), "Permissions denied", Toast.LENGTH_SHORT).show();
}
}
});
// Camera launcher
cameraLauncher = registerForActivityResult(
new ActivityResultContracts.StartActivityForResult(),
result -> {
if (result.getResultCode() == getActivity().RESULT_OK && photoUri != null) {
imageView.setImageURI(photoUri);
viewModel.setSelectedImageUri(photoUri);
}
});
// Gallery launcher
galleryLauncher = registerForActivityResult(
new ActivityResultContracts.GetContent(),
uri -> {
if (uri != null) {
imageView.setImageURI(uri);
viewModel.setSelectedImageUri(uri);
}
});
}
/**
* Check permissions and open camera
*/
private void checkPermissionAndOpenCamera() {
String[] requiredPermissions = getRequiredCameraPermissions();
if (hasPermissions(requiredPermissions)) {
openCamera();
} else {
pendingAction = PENDING_CAMERA;
requestMultiplePermissionsLauncher.launch(requiredPermissions);
}
}
/**
* Check permissions and open gallery
*/
private void checkPermissionAndOpenGallery() {
String[] requiredPermissions = getRequiredGalleryPermissions();
if (hasPermissions(requiredPermissions)) {
openGallery();
} else {
pendingAction = PENDING_GALLERY;
requestMultiplePermissionsLauncher.launch(requiredPermissions);
}
}
/**
* Get required permissions for camera
*/
private String[] getRequiredCameraPermissions() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
// Android 13+ - Camera only (storage permission not needed for camera)
return new String[]{Manifest.permission.CAMERA};
} else {
// Android 12 and below - Camera + storage for saving photo
return new String[]{
Manifest.permission.CAMERA,
Manifest.permission.WRITE_EXTERNAL_STORAGE
};
}
}
/**
* Get required permissions for gallery
*/
private String[] getRequiredGalleryPermissions() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
// Android 13+ uses READ_MEDIA_IMAGES
return new String[]{Manifest.permission.READ_MEDIA_IMAGES};
} else {
// Android 12 and below use READ_EXTERNAL_STORAGE
return new String[]{Manifest.permission.READ_EXTERNAL_STORAGE};
}
}
/**
* Check if all permissions are granted
*/
private boolean hasPermissions(String[] permissions) {
for (String permission : permissions) {
if (ContextCompat.checkSelfPermission(requireContext(), permission)
!= PackageManager.PERMISSION_GRANTED) {
return false;
}
}
return true;
}
private void openCamera() {
try {
File photoFile = createImageFile();
photoUri = FileProvider.getUriForFile(
requireContext(),
requireContext().getPackageName() + ".provider",
photoFile
);
Intent cameraIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
cameraIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoUri);
cameraLauncher.launch(cameraIntent);
} catch (IOException e) {
e.printStackTrace();
Toast.makeText(getContext(), "Error creating file", Toast.LENGTH_SHORT).show();
}
}
private void openGallery() {
galleryLauncher.launch("image/*");
}
private File createImageFile() throws IOException {
// Create an image file name
String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss", Locale.getDefault()).format(new Date());
String imageFileName = "JPEG_" + timeStamp + "_";
File storageDir = requireContext().getExternalFilesDir(null);
File image = File.createTempFile(
imageFileName, /* prefix */
".jpg", /* suffix */
storageDir /* directory */
);
return image;
}
}

View File

@ -0,0 +1,134 @@
package com.utopiaindustries.qc_android.fragments;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Button;
import android.widget.EditText;
import android.widget.TextView;
import android.widget.Toast;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.fragment.app.Fragment;
import androidx.lifecycle.ViewModelProvider;
import com.utopiaindustries.qc_android.R;
import com.utopiaindustries.qc_android.viewmodels.DataEntryViewModel;
public class FragmentQcTerryThree extends Fragment {
private EditText edtCompany, edtPosition, edtSalary;
private Button btnSubmit;
private TextView tvSummary;
private DataEntryViewModel viewModel;
@Nullable
@Override
public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container,
@Nullable Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.fragment_qc_terry_three, container, false);
edtCompany = view.findViewById(R.id.edtCompany);
edtPosition = view.findViewById(R.id.edtPosition);
edtSalary = view.findViewById(R.id.edtSalary);
btnSubmit = view.findViewById(R.id.btnSubmit);
tvSummary = view.findViewById(R.id.tvSummary);
return view;
}
@Override
public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) {
super.onViewCreated(view, savedInstanceState);
viewModel = new ViewModelProvider(requireActivity()).get(DataEntryViewModel.class);
// Load existing data
if (viewModel.getCompany().getValue() != null) {
edtCompany.setText(viewModel.getCompany().getValue());
}
if (viewModel.getPosition().getValue() != null) {
edtPosition.setText(viewModel.getPosition().getValue());
}
// Display summary from previous screens
displaySummary();
btnSubmit.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// Validate input
if (edtCompany.getText().toString().trim().isEmpty()) {
edtCompany.setError("Company is required");
return;
}
if (edtPosition.getText().toString().trim().isEmpty()) {
edtPosition.setError("Position is required");
return;
}
// Save final data
viewModel.setCompany(edtCompany.getText().toString());
viewModel.setPosition(edtPosition.getText().toString());
// Submit data
submitData();
}
});
}
private void displaySummary() {
StringBuilder summary = new StringBuilder();
summary.append("Summary:\n\n");
if (viewModel.getFirstName().getValue() != null &&
!viewModel.getFirstName().getValue().isEmpty()) {
summary.append("Name: ")
.append(viewModel.getFirstName().getValue())
.append(" ")
.append(viewModel.getLastName().getValue())
.append("\n");
}
if (viewModel.getEmail().getValue() != null &&
!viewModel.getEmail().getValue().isEmpty()) {
summary.append("Email: ")
.append(viewModel.getEmail().getValue())
.append("\n");
}
if (viewModel.getAddress().getValue() != null &&
!viewModel.getAddress().getValue().isEmpty()) {
summary.append("Address: ")
.append(viewModel.getAddress().getValue())
.append("\n");
}
if (viewModel.getCity().getValue() != null &&
!viewModel.getCity().getValue().isEmpty()) {
summary.append("City: ")
.append(viewModel.getCity().getValue())
.append("\n");
}
tvSummary.setText(summary.toString());
}
private void submitData() {
// Show success message
Toast.makeText(getActivity(), "Data submitted successfully!", Toast.LENGTH_SHORT).show();
// Update UI
btnSubmit.setText("Submitted ✓");
btnSubmit.setEnabled(false);
btnSubmit.setBackgroundColor(getResources().getColor(android.R.color.darker_gray));
// You would typically:
// 1. Send data to API/backend
// 2. Save to database
// 3. Navigate to success screen
}
}

View File

@ -0,0 +1,509 @@
package com.utopiaindustries.qc_android.fragments;
import android.Manifest;
import android.content.Context;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.graphics.Color;
import android.graphics.drawable.ColorDrawable;
import android.net.Uri;
import android.os.Build;
import android.os.Bundle;
import android.provider.MediaStore;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Button;
import android.widget.EditText;
import android.widget.ImageView;
import android.widget.TextView;
import android.widget.Toast;
import androidx.activity.result.ActivityResultCallback;
import androidx.activity.result.ActivityResultLauncher;
import androidx.activity.result.contract.ActivityResultContracts;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.appcompat.app.AlertDialog;
import androidx.core.content.ContextCompat;
import androidx.core.content.FileProvider;
import androidx.fragment.app.Fragment;
import androidx.lifecycle.ViewModelProvider;
import androidx.navigation.Navigation;
import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView;
import com.utopiaindustries.qc_android.R;
import com.utopiaindustries.qc_android.adapters.CheckPointAdapter;
import com.utopiaindustries.qc_android.models.InspectionCheckPoint;
import com.utopiaindustries.qc_android.viewmodels.DataEntryViewModel;
import java.io.File;
import java.io.IOException;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Objects;
public class FragmentQcTerryTwo extends Fragment implements CheckPointAdapter.OnCheckPointClickListener{
private Button btnNext;
private DataEntryViewModel viewModel;
private CheckPointAdapter adapter;
private List<InspectionCheckPoint> checkPointList = new ArrayList<>();
private RecyclerView recyclerView;
private ActivityResultLauncher<String[]> requestMultiplePermissionsLauncher;
// Activity Result Launchers
private ActivityResultLauncher<Intent> cameraLauncher;
private ActivityResultLauncher<String> galleryLauncher;
private Uri photoUri;
private static final int PENDING_CAMERA = 1;
private static final int PENDING_GALLERY = 2;
private int pendingAction = 0;
@Nullable
@Override
public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container,
@Nullable Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.fragment_qc_terry_two, container, false);
initializeLayouts(view);
return view;
}
public void initializeLayouts(View view) {
recyclerView = view.findViewById(R.id.check_point_recycler_view);
btnNext = view.findViewById(R.id.btnNext);
// Setup RecyclerView
recyclerView.setLayoutManager(new LinearLayoutManager(getActivity()));
adapter = new CheckPointAdapter(getActivity(), checkPointList, this);
recyclerView.setAdapter(adapter);
//loadCheckPoints();
}
private void loadCheckPoints() {
checkPointList.clear();
// Item 1: Using 7-field constructor
List<byte[]> item1Images = createImages("Helmet", 2);
InspectionCheckPoint item1 = new InspectionCheckPoint(
1, // id
"Safety Helmet Check", // title
"Safety", // category
true, // isOkChecked
false, // isNoChecked
"All workers wearing helmets properly", // remarks
item1Images // images
);
checkPointList.add(item1);
// Item 2: NO checked with single image
List<byte[]> item2Images = new ArrayList<>();
item2Images.add(createSampleImageBytes("Fire_Extinguisher"));
InspectionCheckPoint item2 = new InspectionCheckPoint(
2,
"Fire Extinguisher Check",
"Safety",
false,
true,
"Fire extinguisher expired, needs replacement",
item2Images
);
checkPointList.add(item2);
// Item 3: Quality check with multiple images
List<byte[]> item3Images = createImages("Material", 3);
InspectionCheckPoint item3 = new InspectionCheckPoint(
3,
"Material Quality Inspection",
"Quality",
true,
false,
"Material meets all specifications",
item3Images
);
checkPointList.add(item3);
// Item 4: NO status with many images
List<byte[]> item4Images = createImages("Weld_Defect", 5);
InspectionCheckPoint item4 = new InspectionCheckPoint(
4,
"Weld Quality Check",
"Quality",
false,
true,
"Poor weld quality, needs rework",
item4Images
);
checkPointList.add(item4);
// Item 5: Pending (no status selected), no images
InspectionCheckPoint item5 = new InspectionCheckPoint(
5,
"Electrical Panel Inspection",
"Electrical",
false,
false,
"Awaiting electrician's review",
new ArrayList<>() // Empty images list
);
checkPointList.add(item5);
// Item 6: OK checked with one image
List<byte[]> item6Images = new ArrayList<>();
item6Images.add(createSampleImageBytes("Wiring"));
InspectionCheckPoint item6 = new InspectionCheckPoint(
6,
"Wiring Safety Check",
"Electrical",
true,
false,
"All wiring properly insulated and secured",
item6Images
);
checkPointList.add(item6);
// Item 8: NO checked, no remarks
List<byte[]> item8Images = createImages("Missing_Guard", 2);
InspectionCheckPoint item8 = new InspectionCheckPoint(
8,
"Machine Guarding Check",
"Mechanical",
false,
true,
null, // No remarks
item8Images
);
checkPointList.add(item8);
// Item 9: OK with default empty images
InspectionCheckPoint item9 = new InspectionCheckPoint(
9,
"Structural Integrity",
"Civil",
true,
false,
"Structure stable, no cracks observed",
null // Will be converted to empty list in constructor
);
checkPointList.add(item9);
// Add more items using loops
for (int i = 11; i <= 15; i++) {
String[] categories = {"Safety", "Quality", "Electrical", "General"};
String category = categories[(i - 11) % categories.length];
List<byte[]> images = new ArrayList<>();
int imageCount = (i % 3); // 0, 1, or 2 images
for (int j = 0; j < imageCount; j++) {
images.add(createSampleImageBytes("CP" + i + "_Img" + j));
}
boolean isOkChecked = (i % 2 == 0);
boolean isNoChecked = (i % 3 == 0);
InspectionCheckPoint item = new InspectionCheckPoint(
i,
"Check Point " + i,
category,
isOkChecked,
isNoChecked,
"Remarks for check point " + i,
images
);
checkPointList.add(item);
}
adapter.notifyDataSetChanged();
}
private List<byte[]> createImages(String prefix, int count) {
List<byte[]> images = new ArrayList<>();
for (int i = 1; i <= count; i++) {
images.add(createSampleImageBytes(prefix + "_" + i));
}
return images;
}
private byte[] createSampleImageBytes(String imageName) {
String imageData = "SAMPLE_IMAGE[" + imageName + "]_" + System.currentTimeMillis();
return imageData.getBytes();
}
@Override
public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) {
super.onViewCreated(view, savedInstanceState);
viewModel = new ViewModelProvider(requireActivity()).get(DataEntryViewModel.class);
// Load existing data
/*if (viewModel.getAddress().getValue() != null) {
edtAddress.setText(viewModel.getAddress().getValue());
}
if (viewModel.getCity().getValue() != null) {
edtCity.setText(viewModel.getCity().getValue());
}*/
btnNext.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// Save data
/*viewModel.setAddress(edtAddress.getText().toString());
viewModel.setCity(edtCity.getText().toString());*/
// Navigate to next
Navigation.findNavController(v)
.navigate(R.id.action_fragmentTwo_to_fragmentThree);
}
});
}
@Override
public void onImagePickerClicked(int position) {
// Handle image picker click
//openImagePicker(position);
}
@Override
public void onImageClicked(int parentPosition, int imagePosition, Uri imageUri) {
// Handle image click - maybe show in full screen
// showImageFullScreen(imageUri);
}
@Override
public void onImageDeleteClicked(int parentPosition, int imagePosition) {
// Remove image from checkpoint
//adapter.removeImageFromCheckPoint(parentPosition, imagePosition);
}
@Override
public void onDeleteAllImagesClicked(int position) {
}
@Override
public void onRemarksChanged(int position, String remarks) {
// Update remarks in the model
InspectionCheckPoint item = checkPointList.get(position);
item.setRemarks(remarks);
}
@Override
public void onCheckBoxChanged(int position, boolean isOkChecked, boolean isNoChecked) {
// Handle checkbox changes
InspectionCheckPoint item = checkPointList.get(position);
item.setOkChecked(isOkChecked);
item.setNoChecked(isNoChecked);
}
private void openImagePicker(int position) {
// Open camera or gallery
// After getting image URI:
// Uri imageUri = ...;
// adapter.addImageToCheckPoint(position, imageUri);
}
public void alertForPictures(Context con) {
ViewGroup viewGroup = requireActivity().findViewById(android.R.id.content);
TextView dialogCameraBtn, dialogGalleryBtn;
AlertDialog.Builder builder = new AlertDialog.Builder(con);
View view1 = LayoutInflater.from(con).inflate(R.layout.custom_layout_for_image, viewGroup, false);
builder.setCancelable(true);
builder.setView(view1);
dialogCameraBtn = view1.findViewById(R.id.dialog_camera_btn);
dialogGalleryBtn = view1.findViewById(R.id.dialog_gallery_btn);
AlertDialog alertDialog = builder.create();
alertDialog.getWindow().setBackgroundDrawable(new ColorDrawable(Color.TRANSPARENT));
dialogCameraBtn.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
alertDialog.dismiss();
checkPermissionAndOpenCamera();
}
});
dialogGalleryBtn.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
alertDialog.dismiss();
checkPermissionAndOpenGallery();
}
});
alertDialog.show();
}
private void initializeLaunchers() {
requestMultiplePermissionsLauncher = registerForActivityResult(
new ActivityResultContracts.RequestMultiplePermissions(),
new ActivityResultCallback<Map<String, Boolean>>() {
@Override
public void onActivityResult(Map<String, Boolean> result) {
boolean allGranted = true;
for (Boolean granted : result.values()) {
if (!granted) {
allGranted = false;
break;
}
}
if (allGranted) {
// All permissions granted, proceed based on what was requested
if (pendingAction == PENDING_CAMERA) {
openCamera();
} else if (pendingAction == PENDING_GALLERY) {
openGallery();
}
} else {
// Some permissions denied
Toast.makeText(getContext(), "Permissions denied", Toast.LENGTH_SHORT).show();
}
}
});
// Camera launcher
cameraLauncher = registerForActivityResult(
new ActivityResultContracts.StartActivityForResult(),
result -> {
if (result.getResultCode() == getActivity().RESULT_OK && photoUri != null) {
//imageView.setImageURI(photoUri);
viewModel.setSelectedImageUri(photoUri);
}
});
// Gallery launcher
galleryLauncher = registerForActivityResult(
new ActivityResultContracts.GetContent(),
uri -> {
if (uri != null) {
//imageView.setImageURI(uri);
viewModel.setSelectedImageUri(uri);
}
});
}
/**
* Check permissions and open camera
*/
private void checkPermissionAndOpenCamera() {
String[] requiredPermissions = getRequiredCameraPermissions();
if (hasPermissions(requiredPermissions)) {
openCamera();
} else {
pendingAction = PENDING_CAMERA;
requestMultiplePermissionsLauncher.launch(requiredPermissions);
}
}
/**
* Check permissions and open gallery
*/
private void checkPermissionAndOpenGallery() {
String[] requiredPermissions = getRequiredGalleryPermissions();
if (hasPermissions(requiredPermissions)) {
openGallery();
} else {
pendingAction = PENDING_GALLERY;
requestMultiplePermissionsLauncher.launch(requiredPermissions);
}
}
/**
* Get required permissions for camera
*/
private String[] getRequiredCameraPermissions() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
// Android 13+ - Camera only (storage permission not needed for camera)
return new String[]{Manifest.permission.CAMERA};
} else {
// Android 12 and below - Camera + storage for saving photo
return new String[]{
Manifest.permission.CAMERA,
Manifest.permission.WRITE_EXTERNAL_STORAGE
};
}
}
/**
* Get required permissions for gallery
*/
private String[] getRequiredGalleryPermissions() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
// Android 13+ uses READ_MEDIA_IMAGES
return new String[]{Manifest.permission.READ_MEDIA_IMAGES};
} else {
// Android 12 and below use READ_EXTERNAL_STORAGE
return new String[]{Manifest.permission.READ_EXTERNAL_STORAGE};
}
}
/**
* Check if all permissions are granted
*/
private boolean hasPermissions(String[] permissions) {
for (String permission : permissions) {
if (ContextCompat.checkSelfPermission(requireContext(), permission)
!= PackageManager.PERMISSION_GRANTED) {
return false;
}
}
return true;
}
private void openCamera() {
try {
File photoFile = createImageFile();
photoUri = FileProvider.getUriForFile(
requireContext(),
requireContext().getPackageName() + ".provider",
photoFile
);
Intent cameraIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
cameraIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoUri);
cameraLauncher.launch(cameraIntent);
} catch (IOException e) {
e.printStackTrace();
Toast.makeText(getContext(), "Error creating file", Toast.LENGTH_SHORT).show();
}
}
private void openGallery() {
galleryLauncher.launch("image/*");
}
private File createImageFile() throws IOException {
// Create an image file name
String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss", Locale.getDefault()).format(new Date());
String imageFileName = "JPEG_" + timeStamp + "_";
File storageDir = requireContext().getExternalFilesDir(null);
File image = File.createTempFile(
imageFileName, /* prefix */
".jpg", /* suffix */
storageDir /* directory */
);
return image;
}
}

View File

@ -0,0 +1,162 @@
package com.utopiaindustries.qc_android.fragments;
import android.content.Context;
import android.content.SharedPreferences;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;
import android.widget.Toast;
import androidx.appcompat.app.AlertDialog;
import androidx.fragment.app.Fragment;
import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView;
import com.utopiaindustries.qc_android.R;
import com.utopiaindustries.qc_android.apiservice.ApiService;
import com.utopiaindustries.qc_android.apiservice.ApiServiceFactory;
import java.util.ArrayList;
import java.util.Objects;
public class ReportsFragment extends Fragment {//implements SelectReportListener {
RecyclerView recyclerView;
ApiService apiService;
//Store store;
//InspectionReportService inspectionReportService;
TextView emptyReportTextView;
//List<Alert> alertlist = new ArrayList<>();
SharedPreferences sharedPreferences;
SharedPreferences.Editor editor;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.fragment_reports, container, false);
initializeViews(view);
// Inflate the layout for this fragment
return view;
}
private void initializeViews(View view) {
sharedPreferences = requireActivity().getSharedPreferences("login_prefs", Context.MODE_PRIVATE);
editor = sharedPreferences.edit();
//store = Store.getInstance();
apiService = ApiServiceFactory.getApiService();
//inspectionReportService = InspectionReportService.getInstance();
recyclerView = view.findViewById(R.id.reports_recyclerview);
emptyReportTextView = view.findViewById(R.id.empty_report_text);
recyclerView.setLayoutManager(new LinearLayoutManager(getContext()));
//Toast.makeText(getContext(), "Reports Fragment", Toast.LENGTH_LONG).show();
}
/*private void refreshReportData() {
SharedPreferences sharedPreferences = getContext().getSharedPreferences("login_prefs", Context.MODE_PRIVATE);
String username = sharedPreferences.getString("username", null);
apiService.fetchAllReports(username).enqueue(
new Callback<FetchReportsModel>() {
@Override
public void onResponse(Call<FetchReportsModel> call, Response<FetchReportsModel> response) {
if (response.isSuccessful() && response.body() != null) {
String appOpened = sharedPreferences.getString("appOpened", "false");
//Log.e("appOpened", appOpened);
if (response.body().getAlerts() != null
&& !response.body().getAlerts().isEmpty() && appOpened.equalsIgnoreCase("false")) {
alertlist = response.body().getAlerts();
showAlertsDialog(alertlist, getActivity());
editor.putString("appOpened", "true");
editor.apply();
}
//ReportAdapter adapter = new ReportAdapter(response.body().getReports(), );
if (response.body().getReports() != null && !response.body().getReports().isEmpty())
{
ReportAdapter adapter = new ReportAdapter(response.body().getReports());
recyclerView.setAdapter(adapter);
if (response.body().getReports().isEmpty())
emptyReportTextView.setVisibility(View.VISIBLE);
}
} else {
System.out.println("Error");
emptyReportTextView.setVisibility(View.VISIBLE);
}
}
@Override
public void onFailure(Call<FetchReportsModel> call, Throwable t) {
System.out.println(t.getMessage());
emptyReportTextView.setVisibility(View.VISIBLE);
}
}
);
}*/
@Override
public void onResume() {
super.onResume();
//refreshReportData();
}
public void updateData() {
// Your logic to update fragment data
//System.out.println("Reports-Update");
//refreshReportData();
}
/*private void showAlertsDialog(List<Alert> alerts, Context context) {
AlertDialog.Builder builder = new AlertDialog.Builder(context);
View dialogView = LayoutInflater.from(context).inflate(R.layout.dialog_alerts, null);
RecyclerView recyclerView = dialogView.findViewById(R.id.recyclerAlerts);
AlertAdapter adapter = new AlertAdapter(alerts, context);
recyclerView.setLayoutManager(new LinearLayoutManager(context));
recyclerView.setAdapter(adapter);
builder.setView(dialogView);
builder.setPositiveButton("Close", null);
AlertDialog dialog = builder.create();
dialog.setCancelable(false);
dialog.show();
dialog.getButton(AlertDialog.BUTTON_POSITIVE).setOnClickListener(v -> {
LinearLayoutManager layoutManager = (LinearLayoutManager) recyclerView.getLayoutManager();
if (layoutManager != null) {
int lastVisible = layoutManager.findLastCompletelyVisibleItemPosition();
int totalItems = adapter.getItemCount();
if (recyclerView.canScrollVertically(1)) {
// User has not scrolled to bottom yet
Toast.makeText(context, "Please read all instructions before closing", Toast.LENGTH_LONG).show();
} else {
// All good, close the dialog
dialog.dismiss();
}
}
});
}*/
}

View File

@ -0,0 +1,22 @@
package com.utopiaindustries.qc_android.helper;
import android.os.Build;
import androidx.annotation.RequiresApi;
import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;
public class DateTimeUtils {
public static final String HTML5_DATETIME_INPUT_FORMAT_WITH_SECONDS = "yyyy-MM-dd'T'HH:mm:ss";
/**
* format localdatetime into a given format string
*/
@RequiresApi(api = Build.VERSION_CODES.O)
public static String getFormattedDateTimeString(LocalDateTime dateTime, String dateTimeFormat ) {
DateTimeFormatter formatter = DateTimeFormatter.ofPattern( dateTimeFormat );
return dateTime.format( formatter );
}
}

View File

@ -0,0 +1,459 @@
package com.utopiaindustries.qc_android.helper;
import android.content.Context;
import android.content.SharedPreferences;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Matrix;
import android.media.ExifInterface;
import android.net.ConnectivityManager;
import android.net.NetworkInfo;
import android.net.Uri;
import android.os.Handler;
import android.os.Looper;
import android.text.TextUtils;
import android.util.Log;
import android.widget.ArrayAdapter;
import android.widget.AutoCompleteTextView;
import androidx.preference.PreferenceManager;
import com.bumptech.glide.Glide;
import com.google.gson.Gson;
import com.google.gson.reflect.TypeToken;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.lang.reflect.Method;
import java.lang.reflect.Type;
import java.net.HttpURLConnection;
import java.net.URL;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;
import java.util.Locale;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.function.Consumer;
public class Helper {
public static final String project_file = "Quality-Control";
public static final String departmentId = "departId";
public static final String departmentName = "departName";
public static final String locationSiteId = "locationSiteId";
public static final String locationSiteName = "locationSiteName";
public static final String unitId = "unitId";
public static final String unitName = "unitName";
public static final String floorId = "floorId";
public static final String floorName = "floorName";
public static final String logInUser = "LogInUser";
public static final String firstTimeApiCall = "isFirstTimeApiCall";
public static final String homeSite = "Sites";
public static final String superVisor = "SuperVisor";
public static final String recordType = "RecordType";
public static final String observationClass = "observationClass";
public static final String observationSubClass = "observationSubClass";
public static final String hseFloors = "hseFloors";
public static final String hseActivities = "hseActivities";
public static final String hseInjuryTypes = "hseInjuryTypes";
public static final String hseKpis = "hseKpis";
public static final String hseDepartment = "hseDepartment";
public static final String hseIncidentTypes = "hseIncidentTypes";
public static final String hseInjuredBodyPart = "hseInjuredBodyPart";
public static final String hsePtwType = "hsePtwType";
public static final String hseSafetyTrainingTopics = "hseSafetyTrainingTopics";
public static final String hseBuildings = "hseBuildings";
public static final String hseWorkingTeams = "hseWorkingTeams";
public static final String hseInjuries = "hseInjuries";
//Live url for images
//public final static String IMAGES_URL = "https://portal.utopiaindustries.pk/uind/images/";
//Test Url for images
public final static String IMAGES_URL = "http://192.168.90.22:8081/uind/images/";
public static boolean isValidEmail(CharSequence target) {
return !TextUtils.isEmpty(target) && android.util.Patterns.EMAIL_ADDRESS.matcher(target).matches();
}
public static boolean isNetworkConnected(Context context) {
ConnectivityManager connectivityManager = (ConnectivityManager) context.getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo activeNetwork = connectivityManager.getActiveNetworkInfo();
return activeNetwork != null && activeNetwork.isConnectedOrConnecting();
}
static public void setPreferenceObject(Context c, Object modal, String key) {
SharedPreferences appSharedPrefs = PreferenceManager.getDefaultSharedPreferences(c.getApplicationContext());
SharedPreferences.Editor prefsEditor = appSharedPrefs.edit();
Gson gson = new Gson();
String jsonObject = gson.toJson(modal);
prefsEditor.putString(key, jsonObject);
prefsEditor.commit();
}
/*static public QualityControlResponse getPreferenceObjectJson(Context c, String key) {
SharedPreferences appSharedPrefs = PreferenceManager.getDefaultSharedPreferences(c.getApplicationContext());
String json = appSharedPrefs.getString(key, "");
Gson gson = new Gson();
QualityControlResponse selectedUser = gson.fromJson(json, QualityControlResponse.class);
return selectedUser;
}*/
/*static public void saveArrayList(List<ItemModel> list, String key, Context context){
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
SharedPreferences.Editor editor = prefs.edit();
Gson gson = new Gson();
String json = gson.toJson(list);
editor.putString(key, json);
editor.apply();
}*/
/*static public List<ItemModel> getArrayList(String key, Context context){
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
// Check if the key exists
if (!prefs.contains(key)) {
return null; // Return null if the key doesn't exist
}
Gson gson = new Gson();
String json = prefs.getString(key, null);
Type type = new TypeToken<ArrayList<ItemModel>>() {}.getType();
return gson.fromJson(json, type);
}*/
static public void RemoveArrayList(String key, Context context) {
if (context != null) {
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
if (prefs.contains(key)) { // Check if the key exists
SharedPreferences.Editor editor = prefs.edit();
editor.remove(key); // Remove the key-value pair
editor.apply(); // Apply changes
///Log.e("SharedPreferences", "Key '" + key + "' removed successfully.");
} else {
Log.e("SharedPreferences", "Key '" + key + "' does not exist.");
}
}
}
//for department, site, unit, floor
public static <T> void saveList(List<T> list, String key, Context context) {
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
SharedPreferences.Editor editor = prefs.edit();
Gson gson = new Gson();
String json = gson.toJson(list); // Convert the list to JSON
editor.putString(key, json);
editor.apply(); // Save to SharedPreferences
}
public static <T> List<T> getList(String key, Context context, Class<T> clazz) {
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
// Check if the key exists
if (!prefs.contains(key)) {
return new ArrayList<>(); // Return an empty list if the key doesn't exist
}
Gson gson = new Gson();
String json = prefs.getString(key, null);
Type type = TypeToken.getParameterized(ArrayList.class, clazz).getType(); // Use the provided class type
return gson.fromJson(json, type); // Convert JSON back to the list
}
public static Uri correctImageRotationAndGetUri(Context context, String imagePath) {
try {
Bitmap bitmap = BitmapFactory.decodeFile(imagePath);
ExifInterface exif = new ExifInterface(imagePath);
int orientation = exif.getAttributeInt(ExifInterface.TAG_ORIENTATION, ExifInterface.ORIENTATION_NORMAL);
Matrix matrix = new Matrix();
switch (orientation) {
case ExifInterface.ORIENTATION_ROTATE_90: matrix.postRotate(90); break;
case ExifInterface.ORIENTATION_ROTATE_180: matrix.postRotate(180); break;
case ExifInterface.ORIENTATION_ROTATE_270: matrix.postRotate(270); break;
}
Bitmap rotatedBitmap = Bitmap.createBitmap(bitmap, 0, 0, bitmap.getWidth(), bitmap.getHeight(), matrix, true);
// Save to new file
File correctedFile = new File(context.getCacheDir(), "corrected_" + System.currentTimeMillis() + ".jpg");
FileOutputStream fos = new FileOutputStream(correctedFile);
rotatedBitmap.compress(Bitmap.CompressFormat.JPEG, 100, fos);
fos.flush();
fos.close();
return Uri.fromFile(correctedFile);
} catch (Exception e) {
e.printStackTrace();
return Uri.fromFile(new File(imagePath)); // fallback
}
}
public static boolean isNullOrEmpty( String str ) {
return str == null || str.isEmpty();
}
public static void setupDropdown(AutoCompleteTextView textView, ArrayAdapter<?> adapter) {
textView.setAdapter(adapter);
// Always show dropdown when focused
textView.setOnFocusChangeListener((v, hasFocus) -> {
if (hasFocus) {
textView.post(textView::showDropDown);
}
});
// Show dropdown when clicked
textView.setOnClickListener(v -> textView.showDropDown());
// Optional: tweak dropdown position
textView.setDropDownVerticalOffset(10);
}
public static void deleteFile(String filePath) {
File file = new File(filePath);
if (file.exists()) {
if (file.delete()) {
System.out.println("File deleted successfully");
} else {
System.out.println("Failed to delete the file");
}
} else {
System.out.println("File does not exist");
}
}
public static String formatDate(String inputDate) {
// Define the input and output date formats
SimpleDateFormat inputFormat = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss", Locale.US);
SimpleDateFormat outputFormat = new SimpleDateFormat("MMM dd, yyyy hh:mm a", Locale.US);
try {
// Parse the input string into a Date object
Date date = inputFormat.parse(inputDate);
// Format the Date object into the desired output string
return outputFormat.format(date);
} catch (ParseException e) {
e.printStackTrace();
return null; // Return null on parsing error
}
}
public static <T> String findTitleById(List<T> list, int id) {
for (T item : list) {
try {
Method getIdMethod = item.getClass().getMethod("getId");
Method getTitleMethod = item.getClass().getMethod("getTitle");
int itemId = (int) getIdMethod.invoke(item);
if (itemId == id) {
return (String) getTitleMethod.invoke(item);
}
} catch (Exception e) {
e.printStackTrace();
return null;
}
}
return null;
}
public static <T> String findTitleByIdForSupervisor(List<T> list, int id) {
for (T item : list) {
try {
Method getIdMethod = item.getClass().getMethod("getId");
Method getTitleMethod = item.getClass().getMethod("getName");
int itemId = (int) getIdMethod.invoke(item);
if (itemId == id) {
return (String) getTitleMethod.invoke(item);
}
} catch (Exception e) {
e.printStackTrace();
return null;
}
}
return null;
}
public static void getImageBytesWithGlide(Context context, String url, Consumer<byte[]> callback) {
new Thread(() -> {
try {
// Load bitmap using Glide
Bitmap bitmap = Glide.with(context)
.asBitmap()
.load(url)
.submit()
.get();
// Convert bitmap to byte array
ByteArrayOutputStream stream = new ByteArrayOutputStream();
bitmap.compress(Bitmap.CompressFormat.PNG, 100, stream);
byte[] byteArray = stream.toByteArray();
// Return result on main thread
new Handler(Looper.getMainLooper()).post(() -> callback.accept(byteArray));
} catch (Exception e) {
e.printStackTrace();
new Handler(Looper.getMainLooper()).post(() -> callback.accept(null));
}
}).start();
}
public static <T> String findTitlesByIds(List<T> list, String idString) {
List<String> titles = new ArrayList<>();
if (idString == null || idString.trim().isEmpty()) {
return "";
}
String[] idTokens = idString.split(",");
for (String token : idTokens) {
try {
int targetId = Integer.parseInt(token.trim());
for (T item : list) {
Method getIdMethod = item.getClass().getMethod("getId");
Method getTitleMethod = item.getClass().getMethod("getTitle");
int itemId = (int) getIdMethod.invoke(item);
if (itemId == targetId) {
titles.add((String) getTitleMethod.invoke(item));
break;
}
}
} catch (NumberFormatException e) {
Log.e("Invalid ID token: ",""+token);
} catch (Exception e) {
e.printStackTrace();
}
}
StringBuilder sb = new StringBuilder();
for (int i = 0; i < titles.size(); i++) {
sb.append(titles.get(i));
if (i < titles.size() - 1) {
sb.append(", ");
}
}
return sb.toString();
}
public interface MultiImageDownloadCallback {
void onSuccess(List<byte[]> images);
void onError(Exception e);
}
public interface BeforeAfterImageCallback {
void onSuccess(List<byte[]> beforeImages, List<byte[]> afterImages);
void onError(Exception e);
}
/*public static void downloadImagesAsync(List<UploadedDocument> urlList, MultiImageDownloadCallback callback) {
ExecutorService executor = Executors.newSingleThreadExecutor();
executor.execute(() -> {
List<byte[]> imageBytesList = new ArrayList<>();
try {
for (UploadedDocument urlString : urlList) {
try {
URL url = new URL(IMAGES_URL + urlString.getGeneratedFilename());
Log.e("URL: ",""+url);
HttpURLConnection connection = (HttpURLConnection) url.openConnection();
connection.connect();
InputStream inputStream = connection.getInputStream();
ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
byte[] buffer = new byte[1024];
int len;
while ((len = inputStream.read(buffer)) != -1) {
byteArrayOutputStream.write(buffer, 0, len);
}
inputStream.close();
connection.disconnect();
imageBytesList.add(byteArrayOutputStream.toByteArray());
} catch (Exception ex) {
ex.printStackTrace();
}
}
new Handler(Looper.getMainLooper()).post(() -> callback.onSuccess(imageBytesList));
} catch (Exception e) {
new Handler(Looper.getMainLooper()).post(() -> callback.onError(e));
}
});
}*/
/*public static void downloadBeforeAfterImagesAsync(List<UploadedDocument> documents, BeforeAfterImageCallback callback) {
ExecutorService executor = Executors.newSingleThreadExecutor();
executor.execute(() -> {
List<byte[]> beforeImages = new ArrayList<>();
List<byte[]> afterImages = new ArrayList<>();
try {
for (UploadedDocument doc : documents) {
try {
URL url = new URL(IMAGES_URL + doc.getGeneratedFilename());
HttpURLConnection connection = (HttpURLConnection) url.openConnection();
connection.connect();
InputStream inputStream = connection.getInputStream();
ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
byte[] buffer = new byte[1024];
int len;
while ((len = inputStream.read(buffer)) != -1) {
byteArrayOutputStream.write(buffer, 0, len);
}
inputStream.close();
connection.disconnect();
byte[] imageBytes = byteArrayOutputStream.toByteArray();
if ("OBSERVATION_BEFORE".equalsIgnoreCase(doc.getType())) {
beforeImages.add(imageBytes);
} else if ("OBSERVATION_AFTER".equalsIgnoreCase(doc.getType())) {
afterImages.add(imageBytes);
}
} catch (Exception ex) {
ex.printStackTrace();
}
}
new Handler(Looper.getMainLooper()).post(() -> callback.onSuccess(beforeImages, afterImages));
} catch (Exception e) {
new Handler(Looper.getMainLooper()).post(() -> callback.onError(e));
}
});
}*/
}

View File

@ -0,0 +1,82 @@
package com.utopiaindustries.qc_android.helper;
import android.content.Context;
import android.content.SharedPreferences;
public class Preference {
private static final int PREFERENCE_MODE_PRIVATE = 0;
private static String MY_STRING_PREF = "mystringpref";
public static String LOGIN_FILE = "login_prefs";
public static String TEMP_FILE = "temp_prefs";
public static String KEY_PIN_CODE = "pin_code";
public static String KEY_CUSTOMER_ID = "cus_id";
public static String KEY_CUSTOMER_NAME = "cus_name";
public static String KEY_USER_IS_MASTER = "user_is_master";
public static SharedPreferences getPrefs(String nameOfFile, Context context) {
return context.getSharedPreferences(nameOfFile, Context.MODE_PRIVATE);
}
public static boolean containsKey(String fileName, String key, Context context) {
return getPrefs(fileName, context).contains(key);
}
public static String getMyStringPref(String fileName, String key, Context context) {
return getPrefs(fileName,context).getString(key, "default");
}
public static boolean getMyBooleanPref(String fileName, String key, Context context) {
return getPrefs(fileName,context).getBoolean(key,false);
}
public static int getMyIntPref(String fileName, String key, Context context) {
return getPrefs(fileName,context).getInt(key, 0);
}
public static long getMyLongPref(String fileName, String key, Context context) {
return getPrefs(fileName,context).getLong(key, 0);
}
public static void setMyLongPref(String fileName, String key, Context context, long value) {
// perform validation etc..
getPrefs(fileName,context).edit().putLong(key, value).apply();
}
public static void setMyStringPref(String fileName, String key, Context context, String value) {
// perform validation etc..
getPrefs(fileName,context).edit().putString(key, value).apply();
}
public static void setMyBooleanPref(String fileName, String key, Context context, boolean value) {
// perform validation etc..
getPrefs(fileName,context).edit().putBoolean(key, value).apply();
}
public static void setMyIntPref(String fileName, String key, Context context, int value) {
// perform validation etc..
getPrefs(fileName,context).edit().putInt(key, value).apply();
}
public static void remove(String fileName, String key, Context context) {
// perform validation etc..
if (context != null) {
if (getPrefs(fileName,context).contains(key)) {
getPrefs(fileName,context).edit().remove(key).apply();
}
}
}
public static void removeAll(String fileName, Context context) {
// perform validation etc..
getPrefs(fileName,context).edit().clear().apply();
}
}

View File

@ -0,0 +1,24 @@
package com.utopiaindustries.qc_android.helper;
import android.content.Context;
import android.content.res.Resources;
import com.utopiaindustries.qc_android.R;
import java.io.InputStream;
import java.util.Properties;
public class PropertyReader {
public static Properties getProperties( Context context ) {
Properties properties = new Properties();
try {
Resources resources = context.getResources();
InputStream inputStream = resources.openRawResource( R.raw.application );
properties.load( inputStream );
} catch ( Exception e ) {
e.printStackTrace();
}
return properties;
}
}

View File

@ -0,0 +1,69 @@
package com.utopiaindustries.qc_android.helper;
import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
import com.fasterxml.jackson.databind.PropertyNamingStrategy;
import com.fasterxml.jackson.databind.annotation.JsonNaming;
import java.util.List;
@JsonIgnoreProperties( ignoreUnknown = true )
@JsonNaming( PropertyNamingStrategy.SnakeCaseStrategy.class )
public class Release {
private String id;
private String tagName;
private String name;
private List<ReleaseAsset> assets;
public Release() {
}
public String getId() {
return id;
}
public void setId( String id ) {
this.id = id;
}
public String getTagName() {
return tagName;
}
public void setTagName( String tagName ) {
this.tagName = tagName;
}
public String getName() {
return name;
}
public void setName( String name ) {
this.name = name;
}
public List<ReleaseAsset> getAssets() {
return assets;
}
public void setAssets( List<ReleaseAsset> assets ) {
this.assets = assets;
}
public String getDownloadUrl() {
if ( this.assets == null || this.assets.isEmpty() ) {
return null;
}
return this.assets.get( 0 ).getBrowserDownloadUrl();
}
@Override
public String toString() {
return "Release{" +
"id='" + id + '\'' +
", tagName='" + tagName + '\'' +
", name='" + name + '\'' +
", assets=" + assets +
'}';
}
}

View File

@ -0,0 +1,95 @@
package com.utopiaindustries.qc_android.helper;
import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
import com.fasterxml.jackson.databind.PropertyNamingStrategy;
import com.fasterxml.jackson.databind.annotation.JsonDeserialize;
import com.fasterxml.jackson.databind.annotation.JsonNaming;
import com.utopiaindustries.qc_android.utils.jackson.ZonedDateTimeDeserializer;
import java.time.ZonedDateTime;
@JsonIgnoreProperties( ignoreUnknown = true )
@JsonNaming( PropertyNamingStrategy.SnakeCaseStrategy.class )
public class ReleaseAsset {
private String id;
private String name;
private long size;
private long downloadCount;
@JsonDeserialize( using = ZonedDateTimeDeserializer.class )
private ZonedDateTime createdAt;
private String uuid;
private String browserDownloadUrl;
public ReleaseAsset() {
}
public String getId() {
return id;
}
public void setId( String id ) {
this.id = id;
}
public String getName() {
return name;
}
public void setName( String name ) {
this.name = name;
}
public long getSize() {
return size;
}
public void setSize( long size ) {
this.size = size;
}
public long getDownloadCount() {
return downloadCount;
}
public void setDownloadCount( long downloadCount ) {
this.downloadCount = downloadCount;
}
public ZonedDateTime getCreatedAt() {
return createdAt;
}
public void setCreatedAt( ZonedDateTime createdAt ) {
this.createdAt = createdAt;
}
public String getUuid() {
return uuid;
}
public void setUuid( String uuid ) {
this.uuid = uuid;
}
public String getBrowserDownloadUrl() {
return browserDownloadUrl;
}
public void setBrowserDownloadUrl( String browserDownloadUrl ) {
this.browserDownloadUrl = browserDownloadUrl;
}
@Override
public String toString() {
return "ReleaseAsset{" +
"id='" + id + '\'' +
", name='" + name + '\'' +
", size=" + size +
", downloadCount=" + downloadCount +
", createdAt=" + createdAt +
", uuid='" + uuid + '\'' +
", browserDownloadUrl='" + browserDownloadUrl + '\'' +
'}';
}
}

View File

@ -0,0 +1,118 @@
package com.utopiaindustries.qc_android.models;
import android.net.Uri;
import java.io.Serializable;
import java.util.ArrayList;
import java.util.List;
public class InspectionCheckPoint implements Serializable {
private static final long serialVersionUID = 1L;
private long id;
private String title;
private String category;
private boolean isOkChecked;
private boolean isNoChecked;
private String remarks;
private List<byte[]> images;
public InspectionCheckPoint() {
}
public InspectionCheckPoint(long id, String title, String category, boolean isOkChecked, boolean isNoChecked, String remarks, List<byte[]> images) {
this.id = id;
this.title = title;
this.category = category;
this.isOkChecked = isOkChecked;
this.isNoChecked = isNoChecked;
this.remarks = remarks;
this.images = images;
}
public long getId() {
return id;
}
public void setId(long id) {
this.id = id;
}
public String getTitle() {
return title;
}
public void setTitle(String title) {
this.title = title;
}
public String getCategory() {
return category;
}
public void setCategory(String category) {
this.category = category;
}
public boolean isOkChecked() {
return isOkChecked;
}
public void setOkChecked(boolean okChecked) {
isOkChecked = okChecked;
}
public List<byte[]> getImages() {
return images;
}
public void setImages(List<byte[]> images) {
this.images = images;
}
public void addImage(byte[] imageBytes) {
if (this.images == null) {
this.images = new ArrayList<>();
}
this.images.add(imageBytes);
}
public void removeImage(int position) {
if (this.images != null && position < this.images.size()) {
this.images.remove(position);
}
}
public void clearAllImages() {
if (this.images != null) {
this.images.clear();
}
}
public String getRemarks() {
return remarks;
}
public void setRemarks(String remarks) {
this.remarks = remarks;
}
public boolean isNoChecked() {
return isNoChecked;
}
public void setNoChecked(boolean noChecked) {
isNoChecked = noChecked;
}
@Override
public String toString() {
return "InspectionCheckPoint{" +
"id=" + id +
", title='" + title + '\'' +
", category='" + category + '\'' +
'}';
}
}

View File

@ -0,0 +1,19 @@
package com.utopiaindustries.qc_android.utils;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class DatabaseExecutor {
private static final ExecutorService executor = Executors.newSingleThreadExecutor();
public static ExecutorService getExecutor() {
return executor;
}
public static void shutdown() {
if (!executor.isShutdown()) {
executor.shutdown();
}
}
}

View File

@ -0,0 +1,98 @@
package com.utopiaindustries.qc_android.utils;
import android.content.Context;
import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
public class FileUtils {
// Save model object as byte array file
public static String saveObjectToFile(Context context, Object model, String fileName) {
File file = new File(context.getFilesDir(), fileName);
ObjectOutputStream oos = null;
FileOutputStream fos = null;
try {
fos = new FileOutputStream(file);
oos = new ObjectOutputStream(fos);
oos.writeObject(model); // model must implement Serializable
oos.flush();
return file.getAbsolutePath();
} catch (IOException e) {
e.printStackTrace();
return null; // return null if saving failed
} finally {
try {
if (oos != null) oos.close();
if (fos != null) fos.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
/*public static StorageManager loadStorageFromFile(String filePath) {
File file = new File(filePath);
if (!file.exists()) return null;
try (FileInputStream fis = new FileInputStream(file);
ObjectInputStream ois = new ObjectInputStream(fis)) {
return (StorageManager) ois.readObject();
} catch (IOException | ClassNotFoundException e) {
e.printStackTrace();
return null;
}
}*/
private static final Gson gson = new GsonBuilder().create();
/*public static String saveObjectToFileWithGson(Context context, StorageManager storage, String fileName) {
try {
File dir = context.getFilesDir();
File file = new File(dir, fileName);
try (FileWriter writer = new FileWriter(file)) {
gson.toJson(storage, writer); // ✅ Save as JSON
}
return file.getAbsolutePath();
} catch (Exception e) {
e.printStackTrace();
return null;
}
}*/
/*public static StorageManager loadStorageFromFileWithGson(String filePath) {
File file = new File(filePath);
if (!file.exists()) return null;
try (FileReader reader = new FileReader(file)) {
return gson.fromJson(reader, StorageManager.class); // ✅ Deserialize safely
} catch (Exception e) {
e.printStackTrace();
return null;
}
}*/
public static String getAppVersion(Context context) {
try {
return context.getPackageManager()
.getPackageInfo(context.getPackageName(), 0).versionName;
} catch (Exception e) {
e.printStackTrace();
return "0.0"; // fallback
}
}
}

View File

@ -0,0 +1,46 @@
package com.utopiaindustries.qc_android.utils;
import android.app.Dialog;
import android.graphics.Bitmap;
import android.os.Bundle;
import android.view.Window;
import android.view.WindowManager;
import android.widget.ImageView;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.fragment.app.DialogFragment;
import com.utopiaindustries.qc_android.R;
public class FullScreenImageDialog extends DialogFragment {
private Bitmap imageBitmap;
public FullScreenImageDialog(Bitmap imageBitmap) {
this.imageBitmap = imageBitmap;
}
@NonNull
@Override
public Dialog onCreateDialog(@Nullable Bundle savedInstanceState) {
Dialog dialog = new Dialog(requireContext());
dialog.requestWindowFeature(Window.FEATURE_NO_TITLE);
dialog.setContentView(R.layout.dialog_fullscreen_image);
/*ImageView imageView = dialog.findViewById(R.id.fullscreenImageView);
imageView.setImageBitmap(imageBitmap);*/
//imageView.setOnClickListener(v -> dismiss()); // Close on image click
ImageView imageView = dialog.findViewById(R.id.img_view);
imageView.setImageBitmap(imageBitmap);
ImageView btnClose = dialog.findViewById(R.id.btn_close);
btnClose.setOnClickListener(v -> dismiss());
dialog.getWindow().setLayout(
WindowManager.LayoutParams.MATCH_PARENT,
WindowManager.LayoutParams.MATCH_PARENT
);
return dialog;
}
}

View File

@ -0,0 +1,33 @@
package com.utopiaindustries.qc_android.utils;
import android.content.Context;
import android.util.AttributeSet;
import android.view.MotionEvent;
import androidx.viewpager.widget.ViewPager;
/**
* Created by Development on 05-Oct-2021.
*/
public class NonSwipeableViewPager extends ViewPager
{
public NonSwipeableViewPager(Context context) {
super(context);
}
public NonSwipeableViewPager(Context context, AttributeSet attrs) {
super(context, attrs);
}
@Override
public boolean onInterceptTouchEvent(MotionEvent event) {
// Never allow swiping to switch between pages
return false;
}
@Override
public boolean onTouchEvent(MotionEvent event) {
// Never allow swiping to switch between pages
return false;
}
}

View File

@ -0,0 +1,37 @@
package com.utopiaindustries.qc_android.utils;
import android.content.DialogInterface;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.fragment.app.DialogFragment;
import com.utopiaindustries.qc_android.R;
public class ProgressDialogFragment extends DialogFragment {
@Nullable
@Override
public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
return inflater.inflate(R.layout.dialog_progress, container, false);
}
@Override
public void onStart() {
super.onStart();
if (getDialog() != null && getDialog().getWindow() != null) {
getDialog().getWindow().setLayout(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT);
getDialog().getWindow().setBackgroundDrawableResource(android.R.color.transparent);
}
}
@Override
public void onCancel(@NonNull DialogInterface dialog) {
// Prevent the dialog from being canceled when touched outside
setCancelable(false);
}
}

View File

@ -0,0 +1,7 @@
package com.utopiaindustries.qc_android.utils;
public interface SelectListener {
//void onItemClicked(InspectionReportWrapper reportWrapper);
}

View File

@ -0,0 +1,35 @@
package com.utopiaindustries.qc_android.utils.jackson;
import android.os.Build;
import androidx.annotation.RequiresApi;
import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.DeserializationContext;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.deser.std.StdDeserializer;
import com.utopiaindustries.qc_android.helper.DateTimeUtils;
import java.io.IOException;
import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;
public class LocalDateTimeDeserializer extends StdDeserializer<LocalDateTime> {
private static final long serialVersionUID = 1L;
@RequiresApi(api = Build.VERSION_CODES.O)
public LocalDateTimeDeserializer() {
super( LocalDateTime.class );
}
@RequiresApi(api = Build.VERSION_CODES.O)
@Override
public LocalDateTime deserialize( JsonParser parser, DeserializationContext ctxt ) throws IOException, JsonProcessingException {
JsonNode node = parser.getCodec().readTree( parser );
String dateStr = node.asText();
DateTimeFormatter formatter = DateTimeFormatter.ofPattern( DateTimeUtils.HTML5_DATETIME_INPUT_FORMAT_WITH_SECONDS );
return LocalDateTime.parse( dateStr, formatter );
}
}

View File

@ -0,0 +1,25 @@
package com.utopiaindustries.qc_android.utils.jackson;
import android.os.Build;
import androidx.annotation.RequiresApi;
import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.databind.JsonSerializer;
import com.fasterxml.jackson.databind.SerializerProvider;
import com.utopiaindustries.qc_android.helper.DateTimeUtils;
import java.io.IOException;
import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;
@RequiresApi(api = Build.VERSION_CODES.O)
public class LocalDateTimeSerializer extends JsonSerializer<LocalDateTime> {
private static final DateTimeFormatter FORMATTER = DateTimeFormatter.ofPattern( DateTimeUtils.HTML5_DATETIME_INPUT_FORMAT_WITH_SECONDS );
@Override
public void serialize( LocalDateTime value, JsonGenerator gen, SerializerProvider serializers ) throws IOException {
gen.writeString( value.format( FORMATTER ) );
}
}

View File

@ -0,0 +1,35 @@
package com.utopiaindustries.qc_android.utils.jackson;
import android.os.Build;
import androidx.annotation.RequiresApi;
import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.DeserializationContext;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.deser.std.StdDeserializer;
import java.io.IOException;
import java.time.LocalDateTime;
import java.time.ZonedDateTime;
import java.time.format.DateTimeFormatter;
public class ZonedDateTimeDeserializer extends StdDeserializer<ZonedDateTime> {
private static final long serialVersionUID = 1L;
@RequiresApi(api = Build.VERSION_CODES.O)
public ZonedDateTimeDeserializer() {
super( LocalDateTime.class );
}
@RequiresApi(api = Build.VERSION_CODES.O)
@Override
public ZonedDateTime deserialize( JsonParser parser, DeserializationContext ctxt ) throws IOException, JsonProcessingException {
JsonNode node = parser.getCodec().readTree( parser );
String dateStr = node.asText();
DateTimeFormatter formatter = DateTimeFormatter.ISO_DATE_TIME;
return ZonedDateTime.parse( dateStr, formatter );
}
}

View File

@ -0,0 +1,90 @@
package com.utopiaindustries.qc_android.viewmodels;
import android.net.Uri;
import androidx.lifecycle.LiveData;
import androidx.lifecycle.MutableLiveData;
import androidx.lifecycle.ViewModel;
public class DataEntryViewModel extends ViewModel {
// Personal Info
private MutableLiveData<String> firstName = new MutableLiveData<>();
private MutableLiveData<String> lastName = new MutableLiveData<>();
private MutableLiveData<String> email = new MutableLiveData<>();
// Address Info
private MutableLiveData<String> address = new MutableLiveData<>();
private MutableLiveData<String> city = new MutableLiveData<>();
// Employment Info
private MutableLiveData<String> company = new MutableLiveData<>();
private MutableLiveData<String> position = new MutableLiveData<>();
private MutableLiveData<Uri> selectedImageUri = new MutableLiveData<>();
// Getters and Setters
public LiveData<String> getFirstName() {
return firstName;
}
public void setFirstName(String firstName) {
this.firstName.setValue(firstName);
}
public LiveData<String> getLastName() {
return lastName;
}
public void setLastName(String lastName) {
this.lastName.setValue(lastName);
}
public LiveData<String> getEmail() {
return email;
}
public void setEmail(String email) {
this.email.setValue(email);
}
public LiveData<String> getAddress() {
return address;
}
public void setAddress(String address) {
this.address.setValue(address);
}
public LiveData<String> getCity() {
return city;
}
public void setCity(String city) {
this.city.setValue(city);
}
public LiveData<String> getCompany() {
return company;
}
public void setCompany(String company) {
this.company.setValue(company);
}
public LiveData<String> getPosition() {
return position;
}
public void setPosition(String position) {
this.position.setValue(position);
}
public MutableLiveData<Uri> getSelectedImageUri() {
return selectedImageUri;
}
public void setSelectedImageUri(Uri uri) {
selectedImageUri.setValue(uri);
}
}

View File

@ -0,0 +1,278 @@
package com.utopiaindustries.qc_android.viewmodels;
import android.content.Context;
import androidx.annotation.NonNull;
import androidx.lifecycle.LiveData;
import androidx.lifecycle.MutableLiveData;
import androidx.lifecycle.ViewModel;
import com.utopiaindustries.qc_android.apiservice.ApiService;
import com.utopiaindustries.qc_android.apiservice.ApiServiceFactory;
import com.utopiaindustries.qc_android.db.CheckpointRepository;
import com.utopiaindustries.qc_android.models.InspectionCheckPoint;
import java.util.List;
import retrofit2.Call;
import retrofit2.Callback;
import retrofit2.Response;
public class ServiceViewModel extends ViewModel {
/*private MutableLiveData<EmployeeInfoResponse> employeeLiveData;
private MutableLiveData<DailyWageResponse> dailyWageLiveData;
private MutableLiveData<HseResponse> userLiveData;
private MutableLiveData<ReportListResponse> reportListLiveData;
private MutableLiveData<HseSaveResponse> userSaveLiveData;
private final ExecutorService executorService;
private MutableLiveData<HseDraftResponse> userDraftLiveData;*/
private ApiService apiService;
private MutableLiveData<Boolean> isLoading;
private MutableLiveData<Boolean> userLoginLiveData;
private MutableLiveData<String> errorLiveData;
private MutableLiveData<List<InspectionCheckPoint>> checkpointsListLiveData;
CheckpointRepository repository;
public ServiceViewModel() {
/*
userLiveData = new MutableLiveData<>();
reportListLiveData = new MutableLiveData<>();
employeeLiveData = new MutableLiveData<>();
dailyWageLiveData = new MutableLiveData<>();
userSaveLiveData = new MutableLiveData<>();
userDraftLiveData = new MutableLiveData<>();
this.executorService = Executors.newFixedThreadPool(4);*/
apiService = ApiServiceFactory.getApiService();
isLoading = new MutableLiveData<>();
userLoginLiveData = new MutableLiveData<>();
errorLiveData = new MutableLiveData<>();
checkpointsListLiveData = new MutableLiveData<>();
}
/*public LiveData<EmployeeInfoResponse> getEmployeeLiveData()
{
return employeeLiveData;
}*/
/*public LiveData<DailyWageResponse> getDailyWageLiveData()
{
return dailyWageLiveData;
}*/
/*public LiveData<HseResponse> getUserLiveData() {
return userLiveData;
}*/
/*public LiveData<ReportListResponse> getReportListLiveData() {
return reportListLiveData;
}*/
/*public LiveData<HseSaveResponse> getUserSaveLiveData() {
return userSaveLiveData;
}*/
/*public LiveData<HseDraftResponse> getUserDraftLiveData() {
return userDraftLiveData;
}*/
public LiveData<Boolean> getLoadingState() {
return isLoading;
}
public LiveData<String> getErrorMessage() {
return errorLiveData;
}
public void isUserAuthenticated(String username, String password, String[] roles) {
isLoading.setValue(true);
apiService.isUserAuthenticated(username, password, roles).enqueue(new Callback<Boolean>() {
@Override
public void onResponse(@NonNull Call<Boolean> call, @NonNull Response<Boolean> response) {
isLoading.setValue(false);
//Log.e("onResponse-1: ", "Successful: "+response);
if (response.isSuccessful() && response.body() != null) {
//Log.e("onResponse-2: ", "Successful: "+response);
userLoginLiveData.setValue(response.body());
} else {
userLoginLiveData.setValue(false);
}
}
@Override
public void onFailure(@NonNull Call<Boolean> call, @NonNull Throwable t) {
//Log.e("onResponse-2: ", "failed"+t.getMessage());
isLoading.setValue(false);
errorLiveData.setValue(t.getMessage());
}
});
}
/*public void sendUserDraftCount(String username, String dCount) {
isLoading.setValue(true);
apiService.sendDraftCount(username, dCount).enqueue(new Callback<HseDraftResponse>() {
@Override
public void onResponse(@NonNull Call<HseDraftResponse> call, @NonNull Response<HseDraftResponse> response) {
isLoading.setValue(false);
if (response.isSuccessful() && response.body() != null) {
userDraftLiveData.setValue(response.body());
} else {
errorLiveData.setValue(response.message());
}
}
@Override
public void onFailure(@NonNull Call<HseDraftResponse> call, @NonNull Throwable t) {
isLoading.setValue(false);
errorLiveData.setValue(t.getMessage());
}
});
}*/
public void getCheckpointsData(Context context) {
repository = new CheckpointRepository(context);
isLoading.setValue(true);
apiService.fetchCheckPoints().enqueue(new Callback<List<InspectionCheckPoint>>() {
@Override
public void onResponse(Call<List<InspectionCheckPoint>> call, Response<List<InspectionCheckPoint>> response) {
isLoading.setValue(false);
if (response.isSuccessful() && response.body() != null) {
//Log.e("onResponse: ", "Successful");
checkpointsListLiveData.setValue(response.body());
repository.insert( response.body() );
} else {
errorLiveData.setValue(response.message());
}
}
@Override
public void onFailure(Call<List<InspectionCheckPoint>> call, Throwable t) {
//Log.e("onResponse: ", "Fail");
isLoading.setValue(false);
errorLiveData.setValue(t.getMessage());
}
});
}
/*public void getDailyWageWorkerData(String visitorId) {
isLoading.setValue(true);
apiService.getDailyWageData(visitorId).enqueue(new Callback<DailyWageResponse>() {
@Override
public void onResponse(Call<DailyWageResponse> call, Response<DailyWageResponse> response) {
isLoading.setValue(false);
if (response.isSuccessful() && response.body() != null) {
//Log.e("onResponse: ", "Successful");
dailyWageLiveData.setValue(response.body());
} else {
errorLiveData.setValue(response.message());
}
}
@Override
public void onFailure(Call<DailyWageResponse> call, Throwable t) {
//Log.e("onResponse: ", "Fail");
isLoading.setValue(false);
errorLiveData.setValue(t.getMessage());
}
});
}*/
/*public void getHSEData() {
isLoading.setValue(true);
apiService.getHseData().enqueue(new Callback<HseResponse>() {
@Override
public void onResponse(Call<HseResponse> call, Response<HseResponse> response) {
isLoading.setValue(false);
if (response.isSuccessful() && response.body() != null) {
//Log.e("onResponse-HseData: ", "Successful");
userLiveData.setValue(response.body());
} else {
errorLiveData.setValue(response.message());
}
}
@Override
public void onFailure(Call<HseResponse> call, Throwable t) {
//Log.e("onResponse: ", "Fail");
isLoading.setValue(false);
errorLiveData.setValue(t.getMessage());
}
});
}*/
/*public void getHSEReportListData(String userName) {
isLoading.setValue(true);
apiService.getHseReportList(userName).enqueue(new Callback<ReportListResponse>() {
@Override
public void onResponse(Call<ReportListResponse> call, Response<ReportListResponse> response) {
isLoading.setValue(false);
if (response.isSuccessful() && response.body() != null) {
//Log.e("onResponse-HseData: ", "Successful");
reportListLiveData.setValue(response.body());
} else {
errorLiveData.setValue(response.message());
}
}
@Override
public void onFailure(Call<ReportListResponse> call, Throwable t) {
//Log.e("onResponse: ", "Fail");
isLoading.setValue(false);
errorLiveData.setValue(t.getMessage());
}
});
}*/
/*public void saveHSEData(HseReportRequest hseRequestModel) {
isLoading.setValue(true);
// Execute the task in the background
executorService.execute(() -> {
try {
apiService.saveHseReport(hseRequestModel).enqueue(new Callback<HseSaveResponse>() {
@Override
public void onResponse(Call<HseSaveResponse> call, Response<HseSaveResponse> response) {
isLoading.setValue(false);
if (response.isSuccessful() && response.body() != null) {
userSaveLiveData.setValue(response.body());
} else {
errorLiveData.setValue(response.message());
}
}
@Override
public void onFailure(Call<HseSaveResponse> call, Throwable t) {
isLoading.setValue(false);
errorLiveData.setValue(t.getMessage());
}
});
} catch (Exception e) {
errorLiveData.postValue(e.getMessage());
isLoading.postValue(false);
}
});
}*/
public LiveData<Boolean> getLoginUser() {
return userLoginLiveData;
}
public LiveData<List<InspectionCheckPoint>> getCheckpointsList() {
return checkpointsListLiveData;
}
/*public LiveData<HseResponse> getUser() {
return userLiveData;
}*/
public LiveData<String> getError() {
return errorLiveData;
}
}

View File

@ -0,0 +1,6 @@
<?xml version="1.0" encoding="utf-8"?>
<layoutAnimation xmlns:android="http://schemas.android.com/apk/res/android"
android:delay="10%"
android:animation="@android:anim/slide_in_left"
android:interpolator="@android:anim/accelerate_decelerate_interpolator"/>

View File

@ -0,0 +1,11 @@
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:fillAfter="true" >
<alpha
android:duration="50"
android:fromAlpha="0.0"
android:interpolator="@android:anim/accelerate_interpolator"
android:toAlpha="1.0" />
</set>

View File

@ -0,0 +1,11 @@
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:fillAfter="true" >
<alpha
android:duration="1000"
android:fromAlpha="1.0"
android:interpolator="@android:anim/accelerate_interpolator"
android:toAlpha="0.0" />
</set>

View File

@ -0,0 +1,14 @@
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:interpolator="@android:anim/accelerate_decelerate_interpolator"
>
<scale xmlns:android="http://schemas.android.com/apk/res/android"
android:fromXScale="0"
android:toXScale="1"
android:fromYScale="0"
android:toYScale="1"
android:pivotX="50%"
android:pivotY="50%"
android:duration="400"
android:fillAfter="true">
</scale>
</set>

View File

@ -0,0 +1,15 @@
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:interpolator="@android:anim/accelerate_decelerate_interpolator"
>
<scale xmlns:android="http://schemas.android.com/apk/res/android"
android:fromXScale="1"
android:toXScale="0"
android:fromYScale="1"
android:toYScale="0"
android:pivotX="50%"
android:pivotY="50%"
android:duration="400"
android:fillAfter="true">
</scale>
</set>

View File

@ -0,0 +1,8 @@
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android">
<translate
android:fromYDelta="0"
android:toYDelta="100%p"
android:duration="300"
android:interpolator="@android:anim/accelerate_decelerate_interpolator"/>
</set>

View File

@ -0,0 +1,8 @@
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android">
<translate
android:fromYDelta="100%p"
android:toYDelta="0"
android:duration="300"
android:interpolator="@android:anim/accelerate_decelerate_interpolator"/>
</set>

View File

@ -0,0 +1,3 @@
<set xmlns:android="http://schemas.android.com/apk/res/android">
<translate android:fromXDelta="0" android:fromYDelta="-1000" android:duration="2000"/>
</set>

View File

@ -0,0 +1,9 @@
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android">
<translate
android:duration="700"
android:fromYDelta="100%"
android:toYDelta="0%"/>
</set>

View File

@ -0,0 +1,11 @@
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:shareInterpolator="false">
<translate
android:duration="500"
android:fromXDelta="-100%"
android:fromYDelta="0%"
android:toXDelta="0%"
android:toYDelta="0%"/>
</set>

View File

@ -0,0 +1,8 @@
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android">
<translate
android:duration="1000"
android:fromXDelta="-100.0%p"
android:interpolator="@android:anim/decelerate_interpolator"
android:toXDelta="0.0" />
</set>

View File

@ -0,0 +1,11 @@
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:shareInterpolator="false">
<translate
android:duration="100"
android:fromXDelta="100%"
android:fromYDelta="0%"
android:toXDelta="0%"
android:toYDelta="0%"/>
</set>

View File

@ -0,0 +1,8 @@
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android">
<translate
android:duration="3000"
android:fromXDelta="100.0%p"
android:interpolator="@android:anim/decelerate_interpolator"
android:toXDelta="0.0%p" />
</set>

View File

@ -0,0 +1,9 @@
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android">
<translate
android:duration="700"
android:fromYDelta="-100%"
android:toYDelta="0%"/>
</set>

View File

@ -0,0 +1,9 @@
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android">
<translate
android:duration="700"
android:fromYDelta="0%"
android:toYDelta="100%"/>
</set>

View File

@ -0,0 +1,11 @@
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:shareInterpolator="false">
<translate
android:duration="500"
android:fromXDelta="0%"
android:fromYDelta="0%"
android:toXDelta="-100%"
android:toYDelta="0%"/>
</set>

View File

@ -0,0 +1,11 @@
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:shareInterpolator="false">
<translate
android:duration="500"
android:fromXDelta="0%"
android:fromYDelta="0%"
android:toXDelta="100%"
android:toYDelta="0%"/>
</set>

View File

@ -0,0 +1,9 @@
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android">
<translate
android:duration="700"
android:fromYDelta="0%"
android:toYDelta="-100%"/>
</set>

View File

@ -0,0 +1,5 @@
<set xmlns:android="http://schemas.android.com/apk/res/android">
<translate android:fromXDelta="0" android:fromYDelta="0"
android:toYDelta="-2000"
android:duration="1000"/>
</set>

View File

@ -0,0 +1,5 @@
<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
<item android:color="#242424" android:state_selected="true" />
<item android:color="#808080" android:state_selected="false" />
</selector>

View File

@ -0,0 +1,5 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android" android:autoMirrored="true" android:height="24dp" android:tint="#FFFFFF" android:viewportHeight="24" android:viewportWidth="24" android:width="24dp">
<path android:fillColor="@android:color/white" android:pathData="M20,11H7.83l5.59,-5.59L12,4l-8,8 8,8 1.41,-1.41L7.83,13H20v-2z"/>
</vector>

View File

@ -0,0 +1,5 @@
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android" android:shape="oval">
<solid android:color="#80000000"/> <!-- Semi-transparent background -->
<size android:width="40dp" android:height="40dp"/>
</shape>

View File

@ -0,0 +1,9 @@
<shape xmlns:android="http://schemas.android.com/apk/res/android">
<corners android:radius="8dp" />
<solid android:color="@color/theme_color" />
<padding
android:bottom="10dp"
android:left="10dp"
android:right="10dp"
android:top="10dp" />
</shape>

Binary file not shown.

After

Width:  |  Height:  |  Size: 13 KiB

View File

@ -0,0 +1,5 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android" android:height="24dp" android:tint="#000000" android:viewportHeight="24" android:viewportWidth="24" android:width="24dp">
<path android:fillColor="@android:color/black" android:pathData="M19,13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z"/>
</vector>

View File

@ -0,0 +1,5 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android" android:height="24dp" android:tint="#FFFFFF" android:viewportHeight="24" android:viewportWidth="24" android:width="24dp">
<path android:fillColor="@android:color/white" android:pathData="M19,6.41L17.59,5 12,10.59 6.41,5 5,6.41 10.59,12 5,17.59 6.41,19 12,13.41 17.59,19 19,17.59 13.41,12z"/>
</vector>

View File

@ -0,0 +1,5 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android" android:height="24dp" android:tint="#000000" android:viewportHeight="24" android:viewportWidth="24" android:width="24dp">
<path android:fillColor="@android:color/white" android:pathData="M6,19c0,1.1 0.9,2 2,2h8c1.1,0 2,-0.9 2,-2V7H6v12zM19,4h-3.5l-1,-1h-5l-1,1H5v2h14V4z"/>
</vector>

View File

@ -0,0 +1,5 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android" android:height="24dp" android:tint="#D3D3D3" android:viewportHeight="24" android:viewportWidth="24" android:width="24dp">
<path android:fillColor="@android:color/white" android:pathData="M20,4L4,4c-1.1,0 -1.99,0.9 -1.99,2L2,18c0,1.1 0.9,2 2,2h16c1.1,0 2,-0.9 2,-2L22,6c0,-1.1 -0.9,-2 -2,-2zM20,8l-8,5 -8,-5L4,6l8,5 8,-5v2z"/>
</vector>

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.1 KiB

View File

@ -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>

View File

@ -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>

View File

@ -0,0 +1,5 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android" android:height="24dp" android:tint="#000000" android:viewportHeight="24" android:viewportWidth="24" android:width="24dp">
<path android:fillColor="@android:color/white" android:pathData="M18,8h-1L17,6c0,-2.76 -2.24,-5 -5,-5S7,3.24 7,6v2L6,8c-1.1,0 -2,0.9 -2,2v10c0,1.1 0.9,2 2,2h12c1.1,0 2,-0.9 2,-2L20,10c0,-1.1 -0.9,-2 -2,-2zM12,17c-1.1,0 -2,-0.9 -2,-2s0.9,-2 2,-2 2,0.9 2,2 -0.9,2 -2,2zM15.1,8L8.9,8L8.9,6c0,-1.71 1.39,-3.1 3.1,-3.1 1.71,0 3.1,1.39 3.1,3.1v2z"/>
</vector>

View File

@ -0,0 +1,5 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android" android:height="24dp" android:tint="#5B5858" android:viewportHeight="24" android:viewportWidth="24" android:width="24dp">
<path android:fillColor="@android:color/white" android:pathData="M21,10.12h-6.78l2.74,-2.82c-2.73,-2.7 -7.15,-2.8 -9.88,-0.1c-2.73,2.71 -2.73,7.08 0,9.79s7.15,2.71 9.88,0C18.32,15.65 19,14.08 19,12.1h2c0,1.98 -0.88,4.55 -2.64,6.29c-3.51,3.48 -9.21,3.48 -12.72,0c-3.5,-3.47 -3.53,-9.11 -0.02,-12.58s9.14,-3.47 12.65,0L21,3V10.12zM12.5,8v4.25l3.5,2.08l-0.72,1.21L11,13V8H12.5z"/>
</vector>

View File

@ -0,0 +1,5 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android" android:height="24dp" android:tint="#000000" android:viewportHeight="24" android:viewportWidth="24" android:width="24dp">
<path android:fillColor="@android:color/white" android:pathData="M21,19V5c0,-1.1 -0.9,-2 -2,-2H5c-1.1,0 -2,0.9 -2,2v14c0,1.1 0.9,2 2,2h14c1.1,0 2,-0.9 2,-2zM8.5,13.5l2.5,3.01L14.5,12l4.5,6H5l3.5,-4.5z"/>
</vector>

View File

@ -0,0 +1,12 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android"
android:width="24dp"
android:height="24dp"
android:tint="#000000"
android:viewportWidth="24"
android:viewportHeight="24">
<path
android:fillColor="@android:color/white"
android:pathData="M18.32,4.26C16.84,3.05 15.01,2.25 13,2.05v2.02c1.46,0.18 2.79,0.76 3.9,1.62L18.32,4.26zM19.93,11h2.02c-0.2,-2.01 -1,-3.84 -2.21,-5.32L18.31,7.1C19.17,8.21 19.75,9.54 19.93,11zM18.31,16.9l1.43,1.43c1.21,-1.48 2.01,-3.32 2.21,-5.32h-2.02C19.75,14.46 19.17,15.79 18.31,16.9zM13,19.93v2.02c2.01,-0.2 3.84,-1 5.32,-2.21l-1.43,-1.43C15.79,19.17 14.46,19.75 13,19.93zM13,12V7h-2v5H7l5,5l5,-5H13zM11,19.93v2.02c-5.05,-0.5 -9,-4.76 -9,-9.95s3.95,-9.45 9,-9.95v2.02C7.05,4.56 4,7.92 4,12S7.05,19.44 11,19.93z" />
</vector>

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.6 KiB

View File

@ -0,0 +1,7 @@
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android"
android:padding="10dp"
android:shape="rectangle">
<solid android:color="@color/grey_600" />
<corners android:radius="2dp" />
</shape>

View File

@ -0,0 +1,13 @@
<?xml version="1.0" encoding="UTF-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android">
<solid android:color="@color/white" />
<stroke
android:width="2dp"
android:color="@color/grey_400" />
<corners android:radius="10dp" />
<padding
android:bottom="0dp"
android:left="0dp"
android:right="0dp"
android:top="0dp" />
</shape>

View File

@ -0,0 +1,5 @@
<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
<item android:drawable="@color/tab_active_bg" android:state_selected="true" />
<item android:drawable="@android:color/transparent" android:state_selected="false" />
</selector>

View File

@ -0,0 +1,111 @@
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout 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:id="@+id/main"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@color/white"
tools:context=".activities.DashboardActivity">
<androidx.appcompat.widget.Toolbar
android:id="@+id/toolbar"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="@color/theme_color"
android:minHeight="?attr/actionBarSize"
android:theme="?attr/actionBarTheme"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<TextView
android:id="@+id/textView"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Home"
android:textColor="@color/white"
android:textSize="@dimen/_12sdp"
app:layout_constraintBottom_toBottomOf="@+id/toolbar"
app:layout_constraintEnd_toEndOf="@+id/toolbar"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="@+id/toolbar" />
<ImageView
android:id="@+id/img_back"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginStart="10dp"
android:padding="10dp"
app:layout_constraintBottom_toBottomOf="@+id/toolbar"
app:layout_constraintStart_toStartOf="@+id/toolbar"
app:layout_constraintTop_toTopOf="@+id/toolbar"
app:srcCompat="@drawable/arrow_back" />
<RelativeLayout
android:layout_width="0dp"
android:layout_height="0dp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/toolbar">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_alignParentBottom="true"
android:background="@color/grey_200"
android:gravity="right"
android:orientation="horizontal"
android:padding="5dp">
<Button
android:id="@+id/btn_reports_upload"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_margin="5dp"
android:background="@drawable/rounded_btn_login"
android:text="Upload Reports"
app:backgroundTint="@color/grey_500" />
<Button
android:id="@+id/btn_app_update"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_margin="5dp"
android:background="@drawable/rounded_btn_login"
android:text="App Update"
app:backgroundTint="@color/grey_500" />
<Button
android:id="@+id/btn_fetch_product"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_margin="5dp"
android:background="@drawable/rounded_btn_login"
android:text="Fetch"
app:backgroundTint="@color/grey_500" />
<Button
android:id="@+id/btn_logout"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_margin="5dp"
android:background="@drawable/rounded_btn_login"
android:text="Logout"
app:backgroundTint="@color/grey_500" />
<Button
android:id="@+id/btn_create_report"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_margin="5dp"
android:background="@drawable/rounded_btn_login"
android:text="Create"
app:backgroundTint="@color/blue_600" />
</LinearLayout>
</RelativeLayout>
</androidx.constraintlayout.widget.ConstraintLayout>

View File

@ -0,0 +1,119 @@
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout 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:id="@+id/main"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@color/white"
tools:context=".activities.LoginActivity">
<LinearLayout
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_marginStart="1dp"
android:layout_marginEnd="1dp"
android:gravity="center"
android:orientation="vertical"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent">
<ImageView
android:layout_width="200dp"
android:layout_height="200dp"
android:layout_marginBottom="20dp"
android:src="@drawable/ic_launcher" />
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginStart="30dp"
android:layout_marginBottom="20dp"
android:padding="5dp"
android:text="Login Now to Continue"
android:textSize="@dimen/_16sdp" />
<com.google.android.material.textfield.TextInputLayout
style="@style/Widget.MaterialComponents.TextInputLayout.OutlinedBox"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:paddingStart="100dp"
android:paddingEnd="100dp"
android:layout_gravity="center_horizontal">
<EditText
android:id="@+id/tf_email"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="@null"
android:drawableStart="@drawable/ic_email"
android:drawablePadding="5dp"
android:drawableTint="@color/grey_700"
android:padding="20dp"
android:hint="User"
android:inputType="text" />
</com.google.android.material.textfield.TextInputLayout>
<com.google.android.material.textfield.TextInputLayout
style="@style/Widget.MaterialComponents.TextInputLayout.OutlinedBox"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_gravity="center_horizontal"
android:layout_marginTop="20dp"
android:paddingStart="100dp"
android:paddingEnd="100dp"
app:passwordToggleEnabled="true">
<EditText
android:id="@+id/tf_password"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="@null"
android:drawableStart="@drawable/ic_password"
android:drawablePadding="5dp"
android:padding="20dp"
android:drawableTint="@color/grey_700"
android:hint="Password"
android:inputType="textPassword" />
</com.google.android.material.textfield.TextInputLayout>
<Button
android:id="@+id/btn_login"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginStart="100dp"
android:layout_marginEnd="100dp"
android:layout_marginTop="30dp"
android:background="@drawable/rounded_btn_login"
android:padding="5dp"
android:text="Login"
android:textColor="@color/white"
android:textSize="@dimen/_15sdp" />
</LinearLayout>
<TextView
android:id="@+id/txt_version"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginEnd="16dp"
android:layout_marginBottom="28dp"
android:textStyle="bold"
android:text="V-1.0"
android:textSize="@dimen/_11sdp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent" />
<ImageView
android:id="@+id/img_check_update"
android:layout_width="50dp"
android:layout_height="50dp"
android:layout_marginTop="16dp"
android:layout_marginEnd="10dp"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintTop_toTopOf="parent"
app:srcCompat="@drawable/icon_update" />
</androidx.constraintlayout.widget.ConstraintLayout>

View File

@ -0,0 +1,18 @@
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout 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:id="@+id/main"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".activities.QCTerryActivity">
<fragment
android:id="@+id/nav_host_fragment"
android:name="androidx.navigation.fragment.NavHostFragment"
android:layout_width="match_parent"
android:layout_height="match_parent"
app:defaultNavHost="true"
app:navGraph="@navigation/nav_graph" />
</androidx.constraintlayout.widget.ConstraintLayout>

View File

@ -0,0 +1,96 @@
<?xml version="1.0" encoding="utf-8"?>
<androidx.cardview.widget.CardView xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_margin="8dp"
android:layout_marginVertical="4dp"
android:clickable="true"
android:elevation="10dp"
android:focusable="true"
android:padding="4dp"
app:cardBackgroundColor="@color/white">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
android:padding="8dp">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal">
<TextView
android:id="@+id/check_point_title"
android:layout_width="wrap_content"
android:layout_height="50dp"
android:layout_weight="4" />
<CheckBox
android:id="@+id/check_point_ok"
android:layout_width="wrap_content"
android:layout_height="50dp"
android:layout_weight="1"
android:text="OK" />
<CheckBox
android:id="@+id/check_point_no"
android:layout_width="wrap_content"
android:layout_height="50dp"
android:layout_weight="1"
android:text="NO" />
</LinearLayout>
<EditText
android:id="@+id/check_point_remarks"
android:layout_width="match_parent"
android:layout_height="50dp"
android:layout_marginTop="3dp"
android:layout_weight="4"
android:hint="Remarks" />
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:weightSum="1">
<androidx.recyclerview.widget.RecyclerView
android:id="@+id/image_recycler_view"
android:layout_width="0dp"
android:layout_height="150dp"
android:layout_margin="10dp"
android:layout_weight="0.9"
android:orientation="horizontal" />
<LinearLayout
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="0.1"
android:orientation="horizontal">
<ImageButton
android:id="@+id/image_picker"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:background="@null"
android:padding="8dp"
android:src="@drawable/image_picker" />
<ImageButton
android:id="@+id/delete_image"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:background="@null"
android:padding="8dp"
android:src="@drawable/ic_delete" />
</LinearLayout>
</LinearLayout>
</LinearLayout>
</androidx.cardview.widget.CardView>

View File

@ -0,0 +1,87 @@
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_gravity="center">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:layout_marginLeft="10dp"
android:layout_marginRight="10dp"
android:background="@drawable/rounded_white"
android:orientation="vertical"
android:padding="15dp">
<TextView
android:id="@+id/message"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_gravity="center_horizontal"
android:layout_marginTop="15dp"
android:layout_marginBottom="15dp"
android:gravity="center_horizontal"
android:paddingLeft="15dp"
android:paddingRight="15dp"
android:text="Select Image from"
android:textAllCaps="false"
android:textColor="@color/black"
android:textSize="@dimen/_15sdp"
android:textStyle="normal" />
<View
android:layout_width="match_parent"
android:layout_height="1px"
android:layout_marginTop="7dp"
android:background="@color/white" />
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:weightSum="3">
<TextView
android:id="@+id/dialog_camera_btn"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1.495"
android:gravity="center"
android:paddingTop="15dp"
android:paddingBottom="15dp"
android:background="@drawable/custom_button"
android:text="Camera"
android:textAllCaps="false"
android:textColor="@color/white"
android:textSize="@dimen/_14sdp"
android:textStyle="normal" />
<TextView
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="0.05"
android:background="@color/white"
android:paddingTop="15dp"
android:paddingBottom="15dp" />
<TextView
android:id="@+id/dialog_gallery_btn"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1.495"
android:gravity="center"
android:paddingTop="15dp"
android:paddingBottom="15dp"
android:text="Gallery"
android:textAllCaps="false"
android:background="@drawable/custom_button"
android:textColor="@color/white"
android:textSize="@dimen/_14sdp"
android:textStyle="normal" />
</LinearLayout>
</LinearLayout>
</LinearLayout>

View File

@ -0,0 +1,20 @@
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:id="@+id/custom_tab"
android:orientation="vertical"
android:background="@drawable/tab_background_selector">
<TextView
android:id="@+id/txt_name"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:text="text"
android:gravity="center"
android:textStyle="bold"
android:textColor="@color/tab_text_selector"
android:layout_gravity="center"
android:textSize="@dimen/_9sdp" />
</LinearLayout>

View File

@ -0,0 +1,31 @@
<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#80000000">
<!--<ImageView
android:id="@+id/fullscreenImageView"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:scaleType="fitCenter"
android:contentDescription="Full-screen image" />-->
<com.ortiz.touchview.TouchImageView
android:id="@+id/img_view"
android:layout_width="match_parent"
android:layout_height="match_parent" />
<ImageView
android:id="@+id/btn_close"
android:layout_width="40dp"
android:layout_height="40dp"
android:layout_margin="3dp"
android:padding="5dp"
android:scaleType="centerInside"
android:background="@drawable/circle_background"
android:src="@drawable/ic_close"
android:contentDescription="Close"
android:layout_gravity="top|end" />
</FrameLayout>

View File

@ -0,0 +1,15 @@
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#80000000"> <!-- Semi-transparent dark background -->
<!-- Blur effect can be enhanced by using drawable with blur effect-->
<ProgressBar
android:id="@+id/progress_bar"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:indeterminateTint="@color/black"
android:indeterminate="true" />
</RelativeLayout>

View File

@ -0,0 +1,36 @@
<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
xmlns:app="http://schemas.android.com/apk/res-auto"
tools:context=".fragments.DraftFragment">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintHorizontal_bias="0.0"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent"
app:layout_constraintVertical_bias="1.0">
<androidx.recyclerview.widget.RecyclerView
android:id="@+id/draft_recyclerview"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_marginBottom="40dp"
android:padding="16dp"/>
<TextView
android:id="@+id/empty_report_text"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Draft"
android:textAlignment="center"
android:visibility="visible"/>
</LinearLayout>
</FrameLayout>

View File

@ -0,0 +1,109 @@
<?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"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
android:padding="16dp">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginBottom="24dp"
android:text="Personal Information"
android:textSize="20sp"
android:textStyle="bold" />
<com.google.android.material.textfield.TextInputLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="16dp"
android:hint="First Name"
app:errorEnabled="true">
<EditText
android:id="@+id/edtFirstName"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:inputType="textPersonName" />
</com.google.android.material.textfield.TextInputLayout>
<com.google.android.material.textfield.TextInputLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="16dp"
android:hint="Last Name"
app:errorEnabled="true">
<EditText
android:id="@+id/edtLastName"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:inputType="textPersonName" />
</com.google.android.material.textfield.TextInputLayout>
<com.google.android.material.textfield.TextInputLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginBottom="32dp"
android:hint="Email"
app:errorEnabled="true">
<EditText
android:id="@+id/edtEmail"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:inputType="textEmailAddress" />
</com.google.android.material.textfield.TextInputLayout>
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal">
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentStart="true"
android:layout_centerInParent="true"
android:padding="5dp"
android:orientation="vertical"
android:layout_marginBottom="16dp">
<Button
android:id="@+id/btnCamera"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_weight="1"
android:text="Camera" />
<Button
android:id="@+id/btnGallery"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_weight="1"
android:text="Gallery " />
</LinearLayout>
<ImageView
android:id="@+id/imageView"
android:layout_width="200dp"
android:layout_height="200dp"
android:padding="5dp"
android:layout_alignParentEnd="true"
android:background="#f0f0f0"
android:scaleType="centerCrop"
android:layout_marginBottom="24dp"/>
</RelativeLayout>
<Button
android:id="@+id/btnNext"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="16dp"
android:text="Next →" />
</LinearLayout>

Some files were not shown because too many files have changed in this diff Show More