first commit
8
.gitignore
vendored
Normal file
@ -0,0 +1,8 @@
|
||||
*.iml
|
||||
.gradle
|
||||
/local.properties
|
||||
/.idea/workspace.xml
|
||||
/.idea/libraries
|
||||
.DS_Store
|
||||
/build
|
||||
/captures
|
1
.idea/.name
Normal file
@ -0,0 +1 @@
|
||||
UBBAttendance
|
22
.idea/compiler.xml
Normal file
@ -0,0 +1,22 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="CompilerConfiguration">
|
||||
<resourceExtensions />
|
||||
<wildcardResourcePatterns>
|
||||
<entry name="!?*.java" />
|
||||
<entry name="!?*.form" />
|
||||
<entry name="!?*.class" />
|
||||
<entry name="!?*.groovy" />
|
||||
<entry name="!?*.scala" />
|
||||
<entry name="!?*.flex" />
|
||||
<entry name="!?*.kt" />
|
||||
<entry name="!?*.clj" />
|
||||
<entry name="!?*.aj" />
|
||||
</wildcardResourcePatterns>
|
||||
<annotationProcessing>
|
||||
<profile default="true" name="Default" enabled="false">
|
||||
<processorPath useClasspath="true" />
|
||||
</profile>
|
||||
</annotationProcessing>
|
||||
</component>
|
||||
</project>
|
3
.idea/copyright/profiles_settings.xml
Normal file
@ -0,0 +1,3 @@
|
||||
<component name="CopyrightManager">
|
||||
<settings default="" />
|
||||
</component>
|
3
.idea/dictionaries/cromer.xml
Normal file
@ -0,0 +1,3 @@
|
||||
<component name="ProjectDictionaryState">
|
||||
<dictionary name="cromer" />
|
||||
</component>
|
6
.idea/encodings.xml
Normal file
@ -0,0 +1,6 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="Encoding">
|
||||
<file url="PROJECT" charset="UTF-8" />
|
||||
</component>
|
||||
</project>
|
19
.idea/gradle.xml
Normal file
@ -0,0 +1,19 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="GradleSettings">
|
||||
<option name="linkedExternalProjectsSettings">
|
||||
<GradleProjectSettings>
|
||||
<option name="distributionType" value="DEFAULT_WRAPPED" />
|
||||
<option name="externalProjectPath" value="$PROJECT_DIR$" />
|
||||
<option name="gradleJvm" value="1.7" />
|
||||
<option name="modules">
|
||||
<set>
|
||||
<option value="$PROJECT_DIR$" />
|
||||
<option value="$PROJECT_DIR$/attendance-ubb" />
|
||||
</set>
|
||||
</option>
|
||||
<option name="resolveModulePerSourceSet" value="false" />
|
||||
</GradleProjectSettings>
|
||||
</option>
|
||||
</component>
|
||||
</project>
|
36
.idea/misc.xml
Normal file
@ -0,0 +1,36 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="EntryPointsManager">
|
||||
<entry_points version="2.0" />
|
||||
</component>
|
||||
<component name="NullableNotNullManager">
|
||||
<option name="myDefaultNullable" value="android.support.annotation.Nullable" />
|
||||
<option name="myDefaultNotNull" value="android.support.annotation.NonNull" />
|
||||
<option name="myNullables">
|
||||
<value>
|
||||
<list size="4">
|
||||
<item index="0" class="java.lang.String" itemvalue="org.jetbrains.annotations.Nullable" />
|
||||
<item index="1" class="java.lang.String" itemvalue="javax.annotation.Nullable" />
|
||||
<item index="2" class="java.lang.String" itemvalue="edu.umd.cs.findbugs.annotations.Nullable" />
|
||||
<item index="3" class="java.lang.String" itemvalue="android.support.annotation.Nullable" />
|
||||
</list>
|
||||
</value>
|
||||
</option>
|
||||
<option name="myNotNulls">
|
||||
<value>
|
||||
<list size="4">
|
||||
<item index="0" class="java.lang.String" itemvalue="org.jetbrains.annotations.NotNull" />
|
||||
<item index="1" class="java.lang.String" itemvalue="javax.annotation.Nonnull" />
|
||||
<item index="2" class="java.lang.String" itemvalue="edu.umd.cs.findbugs.annotations.NonNull" />
|
||||
<item index="3" class="java.lang.String" itemvalue="android.support.annotation.NonNull" />
|
||||
</list>
|
||||
</value>
|
||||
</option>
|
||||
</component>
|
||||
<component name="ProjectRootManager" version="2" languageLevel="JDK_1_8" default="true" project-jdk-name="1.8" project-jdk-type="JavaSDK">
|
||||
<output url="file://$PROJECT_DIR$/build/classes" />
|
||||
</component>
|
||||
<component name="ProjectType">
|
||||
<option name="id" value="Android" />
|
||||
</component>
|
||||
</project>
|
9
.idea/modules.xml
Normal file
@ -0,0 +1,9 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="ProjectModuleManager">
|
||||
<modules>
|
||||
<module fileurl="file://$PROJECT_DIR$/UBBAttendance.iml" filepath="$PROJECT_DIR$/UBBAttendance.iml" />
|
||||
<module fileurl="file://$PROJECT_DIR$/attendance-ubb/attendance-ubb.iml" filepath="$PROJECT_DIR$/attendance-ubb/attendance-ubb.iml" />
|
||||
</modules>
|
||||
</component>
|
||||
</project>
|
12
.idea/runConfigurations.xml
Normal file
@ -0,0 +1,12 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="RunConfigurationProducerService">
|
||||
<option name="ignoredProducers">
|
||||
<set>
|
||||
<option value="org.jetbrains.plugins.gradle.execution.test.runner.AllInPackageGradleConfigurationProducer" />
|
||||
<option value="org.jetbrains.plugins.gradle.execution.test.runner.TestClassGradleConfigurationProducer" />
|
||||
<option value="org.jetbrains.plugins.gradle.execution.test.runner.TestMethodGradleConfigurationProducer" />
|
||||
</set>
|
||||
</option>
|
||||
</component>
|
||||
</project>
|
15
.navigation/attendance-ubb/raw/main.nvg.xml
Normal file
@ -0,0 +1,15 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<navigationModel
|
||||
ns = "http://schemas.android.com?import=com.android.tools.idea.editors.navigation.model.*">
|
||||
<locations
|
||||
class = "java.util.ArrayList">
|
||||
<statePointEntry>
|
||||
<state
|
||||
class = "com.android.tools.idea.editors.navigation.model.ActivityState"
|
||||
className = "cl.cromer.attendance.SubjectList"/>
|
||||
<point
|
||||
x = "196"
|
||||
y = "52"/>
|
||||
</statePointEntry>
|
||||
</locations>
|
||||
</navigationModel>
|
1
attendance-ubb/.gitignore
vendored
Normal file
@ -0,0 +1 @@
|
||||
/build
|
45
attendance-ubb/build.gradle
Normal file
@ -0,0 +1,45 @@
|
||||
apply plugin: 'com.android.application'
|
||||
|
||||
android {
|
||||
signingConfigs {
|
||||
cromer {
|
||||
keyAlias 'cromer'
|
||||
keyPassword 'K7PRxe4eyeXGr3FeULSapUEqHcKAZgkaGe6HmX7cNd873XBLwR'
|
||||
storeFile file('/home/cromer/cromer_cl/keystore/keystore')
|
||||
storePassword 'K7PRxe4eyeXGr3FeULSapUEqHcKAZgkaGe6HmX7cNd873XBLwR'
|
||||
}
|
||||
}
|
||||
compileSdkVersion 27
|
||||
buildToolsVersion "27"
|
||||
defaultConfig {
|
||||
applicationId "cl.cromer.ubb.attendance"
|
||||
minSdkVersion 14
|
||||
targetSdkVersion 27
|
||||
versionCode 1
|
||||
versionName "1.0"
|
||||
signingConfig signingConfigs.cromer
|
||||
}
|
||||
buildTypes {
|
||||
release {
|
||||
minifyEnabled true
|
||||
shrinkResources true
|
||||
signingConfig signingConfigs.cromer
|
||||
proguardFile '/home/cromer/StudioProjects/UBBAttendance/attendance-ubb/proguard-rules.pro'
|
||||
}
|
||||
debug {
|
||||
debuggable true
|
||||
jniDebuggable true
|
||||
minifyEnabled false
|
||||
}
|
||||
}
|
||||
productFlavors {
|
||||
}
|
||||
}
|
||||
|
||||
dependencies {
|
||||
compile fileTree(include: ['*.jar'], dir: 'libs')
|
||||
compile 'com.android.support:appcompat-v7:27.0.0'
|
||||
compile 'com.android.support:design:27.0.0'
|
||||
compile 'com.android.support:cardview-v7:27.0.0'
|
||||
compile 'com.android.support:recyclerview-v7:27.0.0'
|
||||
}
|
BIN
attendance-ubb/libs/poi-3.14-beta1-20151223.jar
Normal file
27
attendance-ubb/proguard-rules.pro
vendored
Normal file
@ -0,0 +1,27 @@
|
||||
# Add project specific ProGuard rules here.
|
||||
# By default, the flags in this file are appended to flags specified
|
||||
# in /mnt/storage/Android/sdk/tools/proguard/proguard-android.txt
|
||||
# You can edit the include path and order by changing the proguardFiles
|
||||
# directive in build.gradle.
|
||||
#
|
||||
# For more details, see
|
||||
# http://developer.android.com/guide/developing/tools/proguard.html
|
||||
|
||||
# Add any project specific keep options here:
|
||||
|
||||
# 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 *;
|
||||
#}
|
||||
-keep public class org.apache.poi.** {*;}
|
||||
-dontwarn org.apache.poi.**
|
||||
|
||||
-dontwarn android.support.v7.**
|
||||
-keep class android.support.v7.** { *; }
|
||||
-keep interface android.support.v7.** { *; }
|
||||
|
||||
-dontwarn android.support.v4.**
|
||||
-keep class android.support.v4.** { *; }
|
||||
-keep interface android.support.v4.** { *; }
|
123
attendance-ubb/src/main/AndroidManifest.xml
Normal file
@ -0,0 +1,123 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<manifest package="cl.cromer.ubb.attendance"
|
||||
xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
|
||||
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
|
||||
<uses-permission android:name="android.permission.WAKE_LOCK"/>
|
||||
|
||||
<application
|
||||
android:allowBackup="true"
|
||||
android:icon="@mipmap/ic_launcher"
|
||||
android:label="@string/app_name"
|
||||
android:supportsRtl="true"
|
||||
android:theme="@style/UBBTheme">
|
||||
<activity
|
||||
android:name=".SubjectList"
|
||||
android:label="@string/app_name"
|
||||
android:theme="@style/UBBTheme.NoActionBar">
|
||||
<intent-filter>
|
||||
<action android:name="android.intent.action.MAIN"/>
|
||||
|
||||
<category android:name="android.intent.category.LAUNCHER"/>
|
||||
</intent-filter>
|
||||
</activity>
|
||||
<activity
|
||||
android:name=".CourseList"
|
||||
android:label="@string/app_name"
|
||||
android:parentActivityName=".SubjectList"
|
||||
android:theme="@style/UBBTheme">
|
||||
<meta-data
|
||||
android:name="android.support.PARENT_ACTIVITY"
|
||||
android:value="cl.cromer.ubb.attendance.SubjectList"/>
|
||||
</activity>
|
||||
<activity
|
||||
android:name=".ClassList"
|
||||
android:label="@string/app_name"
|
||||
android:parentActivityName=".CourseList"
|
||||
android:theme="@style/UBBTheme">
|
||||
<meta-data
|
||||
android:name="android.support.PARENT_ACTIVITY"
|
||||
android:value="cl.cromer.ubb.attendance.CourseList"/>
|
||||
</activity>
|
||||
<activity
|
||||
android:name=".ImportExcel"
|
||||
android:label="@string/app_name"
|
||||
android:parentActivityName=".SubjectList"
|
||||
android:screenOrientation="portrait"
|
||||
android:theme="@style/UBBTheme">
|
||||
<meta-data
|
||||
android:name="android.support.PARENT_ACTIVITY"
|
||||
android:value="cl.cromer.ubb.attendance.SubjectList"/>
|
||||
</activity>
|
||||
<activity
|
||||
android:name=".ImportPhotos"
|
||||
android:label="@string/app_name"
|
||||
android:parentActivityName=".SubjectList"
|
||||
android:screenOrientation="portrait"
|
||||
android:theme="@style/UBBTheme">
|
||||
<meta-data
|
||||
android:name="android.support.PARENT_ACTIVITY"
|
||||
android:value="cl.cromer.ubb.attendance.SubjectList"/>
|
||||
>
|
||||
</activity>
|
||||
<activity
|
||||
android:name=".TakeAttendance"
|
||||
android:label="@string/app_name"
|
||||
android:parentActivityName=".ClassList"
|
||||
android:screenOrientation="portrait"
|
||||
android:theme="@style/UBBTheme">
|
||||
<meta-data
|
||||
android:name="android.support.PARENT_ACTIVITY"
|
||||
android:value="cl.cromer.ubb.attendance.ClassList"/>
|
||||
</activity>
|
||||
<activity
|
||||
android:name=".CorrectAttendance"
|
||||
android:label="@string/app_name"
|
||||
android:parentActivityName=".ClassList"
|
||||
android:theme="@style/UBBTheme">
|
||||
<meta-data
|
||||
android:name="android.support.PARENT_ACTIVITY"
|
||||
android:value="cl.cromer.ubb.attendance.ClassList"/>
|
||||
</activity>
|
||||
<activity
|
||||
android:name=".LateStudentAttendance"
|
||||
android:label="@string/app_name"
|
||||
android:parentActivityName=".ClassList"
|
||||
android:theme="@style/UBBTheme">
|
||||
<meta-data
|
||||
android:name="android.support.PARENT_ACTIVITY"
|
||||
android:value="cl.cromer.ubb.attendance.ClassList"/>
|
||||
</activity>
|
||||
<activity
|
||||
android:name=".ClassReport"
|
||||
android:label="@string/app_name"
|
||||
android:parentActivityName=".SubjectList"
|
||||
android:screenOrientation="portrait"
|
||||
android:theme="@style/UBBTheme">
|
||||
<meta-data
|
||||
android:name="android.support.PARENT_ACTIVITY"
|
||||
android:value="cl.cromer.ubb.attendance.SubjectList"/>
|
||||
</activity>
|
||||
<activity
|
||||
android:name=".About"
|
||||
android:label="@string/app_name"
|
||||
android:parentActivityName=".SubjectList"
|
||||
android:screenOrientation="portrait"
|
||||
android:theme="@style/UBBTheme">
|
||||
<meta-data
|
||||
android:name="android.support.PARENT_ACTIVITY"
|
||||
android:value="cl.cromer.ubb.attendance.SubjectList"/>
|
||||
</activity>
|
||||
<activity
|
||||
android:name=".CourseReport"
|
||||
android:label="@string/app_name"
|
||||
android:parentActivityName=".SubjectList"
|
||||
android:screenOrientation="portrait"
|
||||
android:theme="@style/UBBTheme">
|
||||
<meta-data
|
||||
android:name="android.support.PARENT_ACTIVITY"
|
||||
android:value="cl.cromer.ubb.attendance.SubjectList"/>
|
||||
</activity>
|
||||
</application>
|
||||
|
||||
</manifest>
|
@ -0,0 +1,44 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.content.Intent;
|
||||
import android.net.Uri;
|
||||
import android.os.Bundle;
|
||||
import android.support.v7.app.ActionBar;
|
||||
import android.support.v7.app.AppCompatActivity;
|
||||
import android.support.v7.widget.Toolbar;
|
||||
import android.view.View;
|
||||
|
||||
public class About extends AppCompatActivity {
|
||||
|
||||
@Override
|
||||
protected void onCreate(Bundle savedInstanceState) {
|
||||
super.onCreate(savedInstanceState);
|
||||
setContentView(R.layout.activity_about);
|
||||
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
|
||||
setSupportActionBar(toolbar);
|
||||
ActionBar actionBar = getSupportActionBar();
|
||||
if (actionBar != null) {
|
||||
actionBar.setHomeButtonEnabled(true);
|
||||
actionBar.setDisplayHomeAsUpEnabled(true);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onBackPressed() {
|
||||
super.onBackPressed();
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onSupportNavigateUp() {
|
||||
onBackPressed();
|
||||
return true;
|
||||
}
|
||||
|
||||
public void sendEmail(View view) {
|
||||
Intent intent = new Intent(Intent.ACTION_SENDTO);
|
||||
intent.setData(Uri.parse("mailto:"));
|
||||
intent.putExtra(Intent.EXTRA_EMAIL, "chris@cromer.cl");
|
||||
intent.putExtra(Intent.EXTRA_SUBJECT, "Attendance");
|
||||
startActivity(Intent.createChooser(intent, getResources().getString(R.string.about_email)));
|
||||
}
|
||||
}
|
@ -0,0 +1,60 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
public class Attendance extends Student {
|
||||
private int status;
|
||||
// 0 - uninitialized
|
||||
// 1 - present
|
||||
// 2 - justified
|
||||
// 3 - absent
|
||||
// 4 - late
|
||||
|
||||
private int present;
|
||||
private int justified;
|
||||
private int absent;
|
||||
private int late;
|
||||
|
||||
public Attendance(int status, int studentId, String run, String firstName, String secondName, String firstLastName, String secondLastName, int major, int enrolled, String email, byte[] photo) {
|
||||
super(studentId, run, firstName, secondName, firstLastName, secondLastName, major, enrolled, email, photo);
|
||||
this.setStatus(status);
|
||||
}
|
||||
|
||||
protected int getStatus() {
|
||||
return status;
|
||||
}
|
||||
|
||||
protected void setStatus(int status) {
|
||||
this.status = status;
|
||||
}
|
||||
|
||||
protected int getPresent() {
|
||||
return present;
|
||||
}
|
||||
|
||||
protected void setPresent(int present) {
|
||||
this.present = present;
|
||||
}
|
||||
|
||||
protected int getJustified() {
|
||||
return justified;
|
||||
}
|
||||
|
||||
protected void setJustified(int justified) {
|
||||
this.justified = justified;
|
||||
}
|
||||
|
||||
protected int getAbsent() {
|
||||
return absent;
|
||||
}
|
||||
|
||||
protected void setAbsent(int absent) {
|
||||
this.absent = absent;
|
||||
}
|
||||
|
||||
protected int getLate() {
|
||||
return late;
|
||||
}
|
||||
|
||||
protected void setLate(int late) {
|
||||
this.late = late;
|
||||
}
|
||||
}
|
122
attendance-ubb/src/main/java/cl/cromer/ubb/attendance/Class.java
Normal file
@ -0,0 +1,122 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.content.Context;
|
||||
import android.os.Parcel;
|
||||
import android.os.Parcelable;
|
||||
|
||||
import java.text.DateFormat;
|
||||
import java.util.Calendar;
|
||||
|
||||
public class Class extends Course implements Parcelable {
|
||||
private int classId;
|
||||
private long date;
|
||||
|
||||
public Class() {}
|
||||
|
||||
public Class(Course course, long date) {
|
||||
super(course.getCourseId(), course.getCourseSection(), course.getCourseSemester(), course.getYear());
|
||||
this.setDate(date);
|
||||
}
|
||||
|
||||
public Class(Course course, int classId, long date) {
|
||||
super(course.getCourseId(), course.getCourseSection(), course.getCourseSemester(), course.getYear());
|
||||
this.setClassId(classId);
|
||||
this.setDate(date);
|
||||
}
|
||||
|
||||
protected void setClass(Class classObject) {
|
||||
this.setClassId(classObject.getClassId());
|
||||
this.setDate(classObject.getDate());
|
||||
}
|
||||
|
||||
protected int getClassId() {
|
||||
return classId;
|
||||
}
|
||||
|
||||
protected void setClassId(int classId) {
|
||||
this.classId = classId;
|
||||
}
|
||||
|
||||
protected long getDate() {
|
||||
return date;
|
||||
}
|
||||
|
||||
protected void setDate(long date) {
|
||||
// Remove the time, I only need the date
|
||||
Calendar calendar = Calendar.getInstance();
|
||||
calendar.setTimeInMillis(date);
|
||||
calendar.set(Calendar.HOUR_OF_DAY, 0);
|
||||
calendar.set(Calendar.MINUTE, 0);
|
||||
calendar.set(Calendar.SECOND, 0);
|
||||
calendar.set(Calendar.MILLISECOND, 0);
|
||||
date = calendar.getTimeInMillis();
|
||||
|
||||
this.date = date;
|
||||
}
|
||||
|
||||
protected String getFormattedDate(Context context) {
|
||||
Calendar calendar = Calendar.getInstance();
|
||||
calendar.setTimeInMillis(date);
|
||||
DateFormat dateFormat = DateFormat.getDateInstance(DateFormat.LONG, context.getResources().getConfiguration().locale);
|
||||
return dateFormat.format(calendar.getTime());
|
||||
}
|
||||
|
||||
protected String getFormattedShortDate(Context context) {
|
||||
Calendar calendar = Calendar.getInstance();
|
||||
calendar.setTimeInMillis(date);
|
||||
DateFormat dateFormat = DateFormat.getDateInstance(DateFormat.SHORT, context.getResources().getConfiguration().locale);
|
||||
return dateFormat.format(calendar.getTime());
|
||||
}
|
||||
|
||||
// Parcelable
|
||||
@Override
|
||||
public int describeContents() {
|
||||
//Must be overridden, but I don't need it.
|
||||
return 0;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void writeToParcel(Parcel out, int flags) {
|
||||
out.writeInt(majorId);
|
||||
out.writeString(majorName);
|
||||
out.writeInt(majorCode);
|
||||
out.writeInt(subjectId);
|
||||
out.writeInt(subjectCode);
|
||||
out.writeString(subjectName);
|
||||
out.writeInt(courseId);
|
||||
out.writeInt(courseSection);
|
||||
out.writeInt(courseSemester);
|
||||
out.writeInt(courseYear);
|
||||
out.writeInt(classId);
|
||||
out.writeLong(date);
|
||||
}
|
||||
|
||||
private void readFromParcel(Parcel in) {
|
||||
majorId = in.readInt();
|
||||
majorName = in.readString();
|
||||
majorCode = in.readInt();
|
||||
subjectId = in.readInt();
|
||||
subjectCode = in.readInt();
|
||||
subjectName = in.readString();
|
||||
courseId = in.readInt();
|
||||
courseSection = in.readInt();
|
||||
courseSemester = in.readInt();
|
||||
courseYear = in.readInt();
|
||||
classId = in.readInt();
|
||||
date = in.readLong();
|
||||
}
|
||||
|
||||
private Class(Parcel in) {
|
||||
readFromParcel(in);
|
||||
}
|
||||
|
||||
public static final Parcelable.Creator<Class> CREATOR = new Parcelable.Creator<Class>() {
|
||||
public Class createFromParcel(Parcel in) {
|
||||
return new Class(in);
|
||||
}
|
||||
|
||||
public Class[] newArray(int size) {
|
||||
return new Class[size];
|
||||
}
|
||||
};
|
||||
}
|
1056
attendance-ubb/src/main/java/cl/cromer/ubb/attendance/ClassList.java
Normal file
@ -0,0 +1,77 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.content.Context;
|
||||
import android.support.v7.widget.CardView;
|
||||
import android.support.v7.widget.RecyclerView;
|
||||
import android.view.LayoutInflater;
|
||||
import android.view.View;
|
||||
import android.view.ViewGroup;
|
||||
import android.widget.TextView;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
public class ClassListAdapter extends RecyclerView.Adapter<ClassListAdapter.ClassViewHolder> {
|
||||
|
||||
private List<Class> classes;
|
||||
private Context context;
|
||||
|
||||
public ClassListAdapter(List<Class> classes, Context context) {
|
||||
this.classes = classes;
|
||||
this.context = context;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getItemCount() {
|
||||
return classes.size();
|
||||
}
|
||||
|
||||
@Override
|
||||
public ClassViewHolder onCreateViewHolder(ViewGroup viewGroup, int i) {
|
||||
View view = LayoutInflater.from(viewGroup.getContext()).inflate(R.layout.view_class_list_card, viewGroup, false);
|
||||
return new ClassViewHolder(view);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onBindViewHolder(ClassViewHolder classViewHolder, int i) {
|
||||
classViewHolder.classDate.setText(String.valueOf(classes.get(i).getFormattedDate(context)));
|
||||
classViewHolder.classId.setText(String.valueOf(classes.get(i).getClassId()));
|
||||
classViewHolder.itemView.setLongClickable(true);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onAttachedToRecyclerView(RecyclerView recyclerView) {
|
||||
super.onAttachedToRecyclerView(recyclerView);
|
||||
}
|
||||
|
||||
public void addClass(Class classObject) {
|
||||
classes.add(getItemCount(), classObject);
|
||||
notifyItemInserted(getItemCount());
|
||||
}
|
||||
|
||||
public void updateClass(int index, Class classObject) {
|
||||
classes.set(index, classObject);
|
||||
notifyItemChanged(index);
|
||||
}
|
||||
|
||||
public Class getClass(int index) {
|
||||
return classes.get(index);
|
||||
}
|
||||
|
||||
public void deleteClass(Class classObject) {
|
||||
classes.remove(classObject);
|
||||
}
|
||||
|
||||
public static class ClassViewHolder extends RecyclerView.ViewHolder {
|
||||
public CardView cardView;
|
||||
public TextView classDate;
|
||||
public TextView classId;
|
||||
|
||||
public ClassViewHolder(View itemView) {
|
||||
super(itemView);
|
||||
cardView = (CardView) itemView.findViewById(R.id.class_card_view);
|
||||
classDate = (TextView) itemView.findViewById(R.id.class_date);
|
||||
classId = (TextView) itemView.findViewById(R.id.class_id);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,968 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.content.ActivityNotFoundException;
|
||||
import android.content.Context;
|
||||
import android.content.DialogInterface;
|
||||
import android.content.Intent;
|
||||
import android.content.res.ColorStateList;
|
||||
import android.database.Cursor;
|
||||
import android.database.sqlite.SQLiteDatabase;
|
||||
import android.net.Uri;
|
||||
import android.os.AsyncTask;
|
||||
import android.os.Bundle;
|
||||
import android.os.Environment;
|
||||
import android.os.Handler;
|
||||
import android.os.PowerManager;
|
||||
import android.support.v4.content.ContextCompat;
|
||||
import android.support.v7.app.AlertDialog;
|
||||
import android.support.v7.app.AppCompatActivity;
|
||||
import android.util.Log;
|
||||
import android.view.LayoutInflater;
|
||||
import android.view.View;
|
||||
import android.widget.AdapterView;
|
||||
import android.widget.ArrayAdapter;
|
||||
import android.widget.RelativeLayout;
|
||||
import android.widget.Spinner;
|
||||
import android.widget.Toast;
|
||||
|
||||
import org.apache.poi.hssf.usermodel.HSSFWorkbook;
|
||||
import org.apache.poi.ss.usermodel.Cell;
|
||||
import org.apache.poi.ss.usermodel.CellStyle;
|
||||
import org.apache.poi.ss.usermodel.Row;
|
||||
import org.apache.poi.ss.usermodel.Sheet;
|
||||
import org.apache.poi.ss.usermodel.Workbook;
|
||||
|
||||
import java.io.File;
|
||||
import java.io.FileNotFoundException;
|
||||
import java.io.FileOutputStream;
|
||||
import java.io.IOException;
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
import cl.cromer.ubb.attendance.Progress;
|
||||
import cl.cromer.ubb.attendance.DBSchema.*;
|
||||
|
||||
public class ClassReport extends AppCompatActivity {
|
||||
|
||||
// SQLite database
|
||||
private SQLParser sqlParser = null;
|
||||
private SQLiteDatabase ubbDB = null;
|
||||
|
||||
// Background thread for the database
|
||||
private Thread thread = null;
|
||||
private Handler threadHandler = new Handler();
|
||||
|
||||
// Progress bar
|
||||
private Progress progress = null;
|
||||
|
||||
private AlertDialog choiceDialog = null;
|
||||
private AlertDialog confirmDialog = null;
|
||||
private View reportView;
|
||||
|
||||
private List<Subject> subjects = new ArrayList<>();
|
||||
private List<Course> courses = new ArrayList<>();
|
||||
private List<Class> classes = new ArrayList<>();
|
||||
private List<Attendance> students = new ArrayList<>();
|
||||
|
||||
private Class classObject = new Class();
|
||||
|
||||
private int disabledButtonColor;
|
||||
|
||||
private Context context;
|
||||
|
||||
@Override
|
||||
protected void onCreate(Bundle savedInstanceState) {
|
||||
super.onCreate(savedInstanceState);
|
||||
setContentView(R.layout.activity_class_report);
|
||||
|
||||
this.context = this;
|
||||
|
||||
LayoutInflater inflater = (LayoutInflater) this.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
|
||||
reportView = inflater.inflate(R.layout.view_class_report, new RelativeLayout(this), false);
|
||||
|
||||
// Build the add subject dialog window using the subject view
|
||||
AlertDialog.Builder builder = new AlertDialog.Builder(this);
|
||||
builder.setView(reportView);
|
||||
builder.setPositiveButton(R.string.input_accept, null);
|
||||
builder.setNegativeButton(R.string.input_cancel, new DialogInterface.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(DialogInterface dialog, int which) {
|
||||
finish();
|
||||
}
|
||||
});
|
||||
|
||||
choiceDialog = builder.create();
|
||||
|
||||
choiceDialog.setOnShowListener(new DialogInterface.OnShowListener() {
|
||||
@Override
|
||||
public void onShow(DialogInterface dialogInterface) {
|
||||
ColorStateList colorStateList = choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).getTextColors();
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setEnabled(false);
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setClickable(false);
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_NEGATIVE).setTextColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimaryDark));
|
||||
|
||||
if (colorStateList != null) {
|
||||
disabledButtonColor = colorStateList.getColorForState(new int[] {-android.R.attr.state_enabled}, R.color.colorPrimary);
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
choiceDialog.getWindow().getAttributes().windowAnimations = R.style.DialogAnimation;
|
||||
choiceDialog.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
finish();
|
||||
}
|
||||
});
|
||||
choiceDialog.setCanceledOnTouchOutside(false);
|
||||
|
||||
// Create a progress dialog for slow devices
|
||||
progress = new Progress();
|
||||
progress.show(this, 1);
|
||||
progress.setCancelable(false);
|
||||
|
||||
// Load the SQLite database
|
||||
sqlParser = new SQLParser(this);
|
||||
thread = new Thread(new Runnable() {
|
||||
public void run() {
|
||||
ubbDB = sqlParser.getWritableDatabase();
|
||||
threadHandler.post(new Runnable() {
|
||||
public void run() {
|
||||
databaseLoaded();
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
thread.start();
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
thread.interrupt();
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onDestroy() {
|
||||
super.onDestroy();
|
||||
// We need to get rid of the progressbar if it's showing
|
||||
if (progress != null && progress.isShowing()) {
|
||||
progress.dismiss();
|
||||
}
|
||||
if (choiceDialog != null && choiceDialog.isShowing()) {
|
||||
choiceDialog.dismiss();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onBackPressed() {
|
||||
super.onBackPressed();
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onSupportNavigateUp() {
|
||||
onBackPressed();
|
||||
return true;
|
||||
}
|
||||
|
||||
private void databaseLoaded() {
|
||||
final GetSubjects getSubjects = new GetSubjects(getApplicationContext());
|
||||
getSubjects.execute();
|
||||
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
getSubjects.cancel(true);
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private AdapterView.OnItemSelectedListener subjectListener() {
|
||||
return new AdapterView.OnItemSelectedListener() {
|
||||
@Override
|
||||
public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {
|
||||
courses = new ArrayList<>();
|
||||
classes = new ArrayList<>();
|
||||
if (position == 0) {
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setTextColor(disabledButtonColor);
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setEnabled(false);
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setClickable(false);
|
||||
|
||||
Spinner spinner = (Spinner) reportView.findViewById(R.id.course_spinner);
|
||||
|
||||
List<String> options = new ArrayList<>();
|
||||
options.add(getResources().getString(R.string.report_course));
|
||||
|
||||
ArrayAdapter<String> arrayAdapter = new ArrayAdapter<>(getApplicationContext(), R.layout.view_spinner_selected_item, options);
|
||||
arrayAdapter.setDropDownViewResource(R.layout.view_spinner_dropdown_item);
|
||||
|
||||
spinner.setAdapter(arrayAdapter);
|
||||
spinner.setSelection(0);
|
||||
spinner.setEnabled(false);
|
||||
spinner.setClickable(false);
|
||||
spinner.setOnItemSelectedListener(null);
|
||||
|
||||
spinner = (Spinner) reportView.findViewById(R.id.class_spinner);
|
||||
|
||||
options = new ArrayList<>();
|
||||
options.add(getResources().getString(R.string.report_class));
|
||||
|
||||
arrayAdapter = new ArrayAdapter<>(getApplicationContext(), R.layout.view_spinner_selected_item, options);
|
||||
arrayAdapter.setDropDownViewResource(R.layout.view_spinner_dropdown_item);
|
||||
|
||||
spinner.setAdapter(arrayAdapter);
|
||||
spinner.setSelection(0);
|
||||
spinner.setEnabled(false);
|
||||
spinner.setClickable(false);
|
||||
spinner.setOnItemSelectedListener(null);
|
||||
}
|
||||
else {
|
||||
progress = new Progress();
|
||||
progress.show(context, 1);
|
||||
progress.setCancelable(false);
|
||||
|
||||
final int thePosition = position - 1;
|
||||
|
||||
classObject.setSubject(subjects.get(thePosition));
|
||||
|
||||
sqlParser = new SQLParser(getApplicationContext());
|
||||
thread = new Thread(new Runnable() {
|
||||
public void run() {
|
||||
ubbDB = sqlParser.getWritableDatabase();
|
||||
threadHandler.post(new Runnable() {
|
||||
public void run() {
|
||||
|
||||
final GetCourses getCourses = new GetCourses(getApplicationContext());
|
||||
getCourses.execute(thePosition);
|
||||
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
getCourses.cancel(true);
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
thread.start();
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
thread.interrupt();
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onNothingSelected(AdapterView<?> parent) {
|
||||
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
private AdapterView.OnItemSelectedListener courseListener() {
|
||||
return new AdapterView.OnItemSelectedListener() {
|
||||
@Override
|
||||
public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {
|
||||
classes = new ArrayList<>();
|
||||
if (position == 0) {
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setTextColor(disabledButtonColor);
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setEnabled(false);
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setClickable(false);
|
||||
|
||||
Spinner spinner = (Spinner) reportView.findViewById(R.id.class_spinner);
|
||||
|
||||
List<String> options = new ArrayList<>();
|
||||
options.add(getResources().getString(R.string.report_class));
|
||||
|
||||
ArrayAdapter<String> arrayAdapter = new ArrayAdapter<>(getApplicationContext(), R.layout.view_spinner_selected_item, options);
|
||||
arrayAdapter.setDropDownViewResource(R.layout.view_spinner_dropdown_item);
|
||||
|
||||
spinner.setAdapter(arrayAdapter);
|
||||
spinner.setSelection(0);
|
||||
spinner.setEnabled(false);
|
||||
spinner.setClickable(false);
|
||||
spinner.setOnItemSelectedListener(null);
|
||||
}
|
||||
else {
|
||||
progress = new Progress();
|
||||
progress.show(context, 1);
|
||||
progress.setCancelable(false);
|
||||
|
||||
final int thePosition = position - 1;
|
||||
|
||||
classObject.setCourse(courses.get(thePosition));
|
||||
|
||||
sqlParser = new SQLParser(getApplicationContext());
|
||||
thread = new Thread(new Runnable() {
|
||||
public void run() {
|
||||
ubbDB = sqlParser.getWritableDatabase();
|
||||
threadHandler.post(new Runnable() {
|
||||
public void run() {
|
||||
|
||||
final GetClasses getClasses = new GetClasses(getApplicationContext());
|
||||
getClasses.execute(thePosition);
|
||||
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
getClasses.cancel(true);
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
thread.start();
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
thread.interrupt();
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onNothingSelected(AdapterView<?> parent) {
|
||||
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
private AdapterView.OnItemSelectedListener classListener() {
|
||||
return new AdapterView.OnItemSelectedListener() {
|
||||
@Override
|
||||
public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {
|
||||
if (position == 0) {
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setTextColor(disabledButtonColor);
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setEnabled(false);
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setClickable(false);
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setOnClickListener(null);
|
||||
}
|
||||
else {
|
||||
classObject.setClass(classes.get(position - 1));
|
||||
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setTextColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimaryDark));
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setEnabled(true);
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setClickable(true);
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setOnClickListener(acceptListener());
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onNothingSelected(AdapterView<?> parent) {
|
||||
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
private View.OnClickListener acceptListener() {
|
||||
return new View.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(View view) {
|
||||
progress = new Progress();
|
||||
progress.show(context, 1);
|
||||
progress.setCancelable(false);
|
||||
|
||||
choiceDialog.dismiss();
|
||||
|
||||
sqlParser = new SQLParser(getApplicationContext());
|
||||
thread = new Thread(new Runnable() {
|
||||
public void run() {
|
||||
ubbDB = sqlParser.getWritableDatabase();
|
||||
threadHandler.post(new Runnable() {
|
||||
public void run() {
|
||||
|
||||
final GetAttendance getAttendance = new GetAttendance(getApplicationContext());
|
||||
getAttendance.execute();
|
||||
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
getAttendance.cancel(true);
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
thread.start();
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
thread.interrupt();
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
private class GetSubjects extends AsyncTask<Void, Void, Void> {
|
||||
private Context context;
|
||||
private PowerManager.WakeLock wakeLock;
|
||||
|
||||
protected GetSubjects(Context context) {
|
||||
this.context = context;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
super.onPreExecute();
|
||||
// Let's make sure the CPU doesn't go to sleep
|
||||
PowerManager powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, getClass().getName());
|
||||
if (!wakeLock.isHeld()) {
|
||||
wakeLock.acquire();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Void doInBackground(Void... voids) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
// Get subjects by name in ascending order
|
||||
String query = "SELECT " + sqlParser.combineColumnsStrings(DBSubjects.ALL_COLUMNS, DBMajors.ALL_COLUMNS) + " FROM " + DBSubjects.TABLE_NAME + " INNER JOIN (SELECT " + sqlParser.combineColumnsStrings(DBMajors.ALL_COLUMNS) +" FROM " + DBMajors.TABLE_NAME + ") " + DBMajors.TABLE_NAME + " ON " + DBSubjects.TABLE_NAME + "." + DBSubjects.COLUMN_MAJOR + "=" + DBMajors.TABLE_NAME + "." + DBMajors.COLUMN_ID + " ORDER BY " + DBSubjects.COLUMN_NAME + " ASC";
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.d("ClassReport", query);
|
||||
}
|
||||
Cursor cursor = ubbDB.rawQuery(query, null);
|
||||
|
||||
// Iterate through the database rows
|
||||
while (cursor.moveToNext()) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
Major major = new Major(
|
||||
cursor.getInt(cursor.getColumnIndex(DBMajors.COLUMN_ID)),
|
||||
cursor.getString(cursor.getColumnIndex(DBMajors.COLUMN_NAME)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBMajors.COLUMN_CODE)));
|
||||
subjects.add(
|
||||
new Subject(
|
||||
major,
|
||||
cursor.getInt(cursor.getColumnIndex(DBSubjects.COLUMN_ID)),
|
||||
cursor.getString(cursor.getColumnIndex(DBSubjects.COLUMN_NAME)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBSubjects.COLUMN_CODE))
|
||||
)
|
||||
);
|
||||
}
|
||||
cursor.close();
|
||||
|
||||
// Close the database connection
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Void result) {
|
||||
// Release the kraken errr wakelock
|
||||
List<String> options = new ArrayList<>();
|
||||
options.add(getResources().getString(R.string.report_subject));
|
||||
for (Subject subject : subjects) {
|
||||
options.add(subject.getSubjectName());
|
||||
}
|
||||
|
||||
Spinner spinner = (Spinner) reportView.findViewById(R.id.subject_spinner);
|
||||
|
||||
ArrayAdapter<String> arrayAdapter = new ArrayAdapter<>(getApplicationContext(), R.layout.view_spinner_selected_item, options);
|
||||
arrayAdapter.setDropDownViewResource(R.layout.view_spinner_dropdown_item);
|
||||
|
||||
spinner.setAdapter(arrayAdapter);
|
||||
spinner.setSelection(0);
|
||||
|
||||
spinner.setOnItemSelectedListener(subjectListener());
|
||||
progress.dismiss();
|
||||
wakeLock.release();
|
||||
choiceDialog.show();
|
||||
}
|
||||
}
|
||||
|
||||
private class GetCourses extends AsyncTask<Integer, Void, Void> {
|
||||
private Context context;
|
||||
private PowerManager.WakeLock wakeLock;
|
||||
|
||||
protected GetCourses(Context context) {
|
||||
this.context = context;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
super.onPreExecute();
|
||||
// Let's make sure the CPU doesn't go to sleep
|
||||
PowerManager powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, getClass().getName());
|
||||
if (!wakeLock.isHeld()) {
|
||||
wakeLock.acquire();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Void doInBackground(Integer... position) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
Cursor cursor = ubbDB.query(
|
||||
DBCourses.TABLE_NAME,
|
||||
DBCourses.ALL_COLUMNS,
|
||||
DBCourses.COLUMN_SUBJECT + "=" + subjects.get(position[0]).getSubjectId(),
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
DBCourses.COLUMN_YEAR + " DESC, " + DBCourses.COLUMN_SEMESTER + " DESC, " + DBCourses.COLUMN_SECTION + " ASC",
|
||||
null);
|
||||
|
||||
// Iterate through the database rows
|
||||
while (cursor.moveToNext()) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
courses.add(
|
||||
new Course(
|
||||
subjects.get(position[0]),
|
||||
cursor.getInt(cursor.getColumnIndex(DBCourses.COLUMN_ID)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBCourses.COLUMN_SECTION)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBCourses.COLUMN_SEMESTER)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBCourses.COLUMN_YEAR))
|
||||
)
|
||||
);
|
||||
}
|
||||
cursor.close();
|
||||
|
||||
// Close the database connection
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Void result) {
|
||||
// Release the kraken errr wakelock
|
||||
List<String> options = new ArrayList<>();
|
||||
options.add(getResources().getString(R.string.report_course));
|
||||
for (Course course : courses) {
|
||||
options.add(course.getYear() + "-" + course.getCourseSemester() + " " + getResources().getString(R.string.report_section) + " " + course.getCourseSection());
|
||||
}
|
||||
|
||||
Spinner spinner = (Spinner) reportView.findViewById(R.id.course_spinner);
|
||||
|
||||
ArrayAdapter<String> arrayAdapter = new ArrayAdapter<>(getApplicationContext(), R.layout.view_spinner_selected_item, options);
|
||||
arrayAdapter.setDropDownViewResource(R.layout.view_spinner_dropdown_item);
|
||||
|
||||
spinner.setAdapter(arrayAdapter);
|
||||
spinner.setSelection(0);
|
||||
spinner.setEnabled(true);
|
||||
spinner.setClickable(true);
|
||||
|
||||
spinner.setOnItemSelectedListener(courseListener());
|
||||
progress.dismiss();
|
||||
wakeLock.release();
|
||||
}
|
||||
}
|
||||
|
||||
private class GetClasses extends AsyncTask<Integer, Void, Void> {
|
||||
private Context context;
|
||||
private PowerManager.WakeLock wakeLock;
|
||||
|
||||
protected GetClasses(Context context) {
|
||||
this.context = context;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
super.onPreExecute();
|
||||
// Let's make sure the CPU doesn't go to sleep
|
||||
PowerManager powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, getClass().getName());
|
||||
if (!wakeLock.isHeld()) {
|
||||
wakeLock.acquire();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Void doInBackground(Integer... position) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
Cursor cursor = ubbDB.query(
|
||||
DBClasses.TABLE_NAME,
|
||||
DBClasses.ALL_COLUMNS,
|
||||
DBClasses.COLUMN_COURSE + "=" + courses.get(position[0]).getCourseId(),
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
DBClasses.COLUMN_DATE + " DESC",
|
||||
null);
|
||||
|
||||
// Iterate through the database rows
|
||||
while (cursor.moveToNext()) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
classes.add(
|
||||
new Class(
|
||||
courses.get(position[0]),
|
||||
cursor.getInt(cursor.getColumnIndex(DBClasses.COLUMN_ID)),
|
||||
cursor.getLong(cursor.getColumnIndex(DBClasses.COLUMN_DATE))
|
||||
)
|
||||
);
|
||||
}
|
||||
cursor.close();
|
||||
|
||||
// Close the database connection
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Void result) {
|
||||
// Release the kraken errr wakelock
|
||||
List<String> options = new ArrayList<>();
|
||||
options.add(getResources().getString(R.string.report_class));
|
||||
for (Class classObject : classes) {
|
||||
options.add(classObject.getFormattedDate(getApplicationContext()));
|
||||
}
|
||||
|
||||
Spinner spinner = (Spinner) reportView.findViewById(R.id.class_spinner);
|
||||
|
||||
ArrayAdapter<String> arrayAdapter = new ArrayAdapter<>(getApplicationContext(), R.layout.view_spinner_selected_item, options);
|
||||
arrayAdapter.setDropDownViewResource(R.layout.view_spinner_dropdown_item);
|
||||
|
||||
spinner.setAdapter(arrayAdapter);
|
||||
spinner.setSelection(0);
|
||||
spinner.setEnabled(true);
|
||||
spinner.setClickable(true);
|
||||
|
||||
spinner.setOnItemSelectedListener(classListener());
|
||||
progress.dismiss();
|
||||
wakeLock.release();
|
||||
}
|
||||
}
|
||||
|
||||
private class GetAttendance extends AsyncTask<Void, Void, Boolean> {
|
||||
private Context context;
|
||||
private PowerManager.WakeLock wakeLock;
|
||||
|
||||
protected GetAttendance(Context context) {
|
||||
this.context = context;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
super.onPreExecute();
|
||||
// Let's make sure the CPU doesn't go to sleep
|
||||
PowerManager powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, getClass().getName());
|
||||
if (!wakeLock.isHeld()) {
|
||||
wakeLock.acquire();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Boolean doInBackground(Void... voids) {
|
||||
if (isCancelled()) {
|
||||
return false;
|
||||
}
|
||||
|
||||
String query = "SELECT " + sqlParser.combineColumnsStrings(DBStudents.ALL_COLUMNS, DBAttendance.ALL_COLUMNS) + " FROM " + DBStudents.TABLE_NAME +
|
||||
" INNER JOIN (SELECT " + sqlParser.combineColumnsStrings(DBCoursesStudents.ALL_COLUMNS) + " FROM " + DBCoursesStudents.TABLE_NAME + " WHERE " + DBCoursesStudents.COLUMN_COURSE + "=" + classObject.getCourseId() + ") AS " + DBCoursesStudents.TABLE_NAME + " ON " + DBStudents.TABLE_NAME + "." + DBStudents.COLUMN_ID + "=" + DBCoursesStudents.TABLE_NAME + "." + DBCoursesStudents.COLUMN_STUDENT +
|
||||
" LEFT OUTER JOIN (SELECT " + sqlParser.combineColumnsStrings(DBAttendance.ALL_COLUMNS) + " FROM " + DBAttendance.TABLE_NAME + " WHERE " + DBAttendance.COLUMN_CLASS + "=" + classObject.getClassId() + ") AS " + DBAttendance.TABLE_NAME + " ON " + DBStudents.TABLE_NAME + "." + DBStudents.COLUMN_ID + "=" + DBAttendance.TABLE_NAME + "." + DBAttendance.COLUMN_STUDENT +
|
||||
" ORDER BY " + DBStudents.COLUMN_FIRST_LAST_NAME + " ASC, " + DBStudents.COLUMN_FIRST_NAME + " ASC";
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.d("ClassReport", query);
|
||||
}
|
||||
Cursor cursor = ubbDB.rawQuery(query, null);
|
||||
|
||||
// Iterate through the database rows
|
||||
while (cursor.moveToNext()) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
students.add(new Attendance(
|
||||
cursor.getInt(cursor.getColumnIndex(DBAttendance.COLUMN_ATTENDANCE)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBStudents.COLUMN_ID)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_RUN)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_FIRST_NAME)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_SECOND_NAME)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_FIRST_LAST_NAME)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_SECOND_LAST_NAME)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBStudents.COLUMN_MAJOR)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBStudents.COLUMN_ENROLLED)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_EMAIL)),
|
||||
cursor.getBlob(cursor.getColumnIndex(DBStudents.COLUMN_PHOTO))
|
||||
));
|
||||
}
|
||||
cursor.close();
|
||||
|
||||
Workbook workbook = new HSSFWorkbook();
|
||||
Sheet sheet = workbook.createSheet(getResources().getString(R.string.app_name));
|
||||
|
||||
Excel excel = new Excel(workbook);
|
||||
excel.createStyles();
|
||||
|
||||
Row row;
|
||||
Cell cell;
|
||||
// Major
|
||||
row = sheet.createRow(0);
|
||||
cell = row.createCell(0);
|
||||
cell.setCellValue(classObject.getMajorName() + " - " + classObject.getMajorCode());
|
||||
cell.setCellStyle(excel.getCellStyle("left"));
|
||||
for (int i = 1; i < 7; i++) {
|
||||
cell = row.createCell(i);
|
||||
if (i == 6) {
|
||||
cell.setCellStyle(excel.getCellStyle("right"));
|
||||
}
|
||||
else {
|
||||
cell.setCellStyle(excel.getCellStyle("middle"));
|
||||
}
|
||||
}
|
||||
|
||||
// Subject
|
||||
row = sheet.createRow(1);
|
||||
cell = row.createCell(0);
|
||||
cell.setCellValue(classObject.getSubjectName() + " - " + classObject.getSubjectCode());
|
||||
cell.setCellStyle(excel.getCellStyle("left"));
|
||||
for (int i = 1; i < 7; i++) {
|
||||
cell = row.createCell(i);
|
||||
if (i == 6) {
|
||||
cell.setCellStyle(excel.getCellStyle("right"));
|
||||
}
|
||||
else {
|
||||
cell.setCellStyle(excel.getCellStyle("middle"));
|
||||
}
|
||||
}
|
||||
|
||||
// Columns
|
||||
row = sheet.createRow(2);
|
||||
|
||||
cell = row.createCell(0);
|
||||
cell.setCellValue(getResources().getString(R.string.report_names));
|
||||
cell.setCellStyle(excel.getCellStyle("blue"));
|
||||
|
||||
cell = row.createCell(1);
|
||||
cell.setCellValue(getResources().getString(R.string.report_first_last_name));
|
||||
cell.setCellStyle(excel.getCellStyle("blue"));
|
||||
|
||||
cell = row.createCell(2);
|
||||
cell.setCellValue(getResources().getString(R.string.report_second_last_name));
|
||||
cell.setCellStyle(excel.getCellStyle("blue"));
|
||||
|
||||
cell = row.createCell(3);
|
||||
cell.setCellValue(getResources().getString(R.string.report_present));
|
||||
cell.setCellStyle(excel.getCellStyle("green"));
|
||||
|
||||
cell = row.createCell(4);
|
||||
cell.setCellValue(getResources().getString(R.string.report_late));
|
||||
cell.setCellStyle(excel.getCellStyle("yellow"));
|
||||
|
||||
cell = row.createCell(5);
|
||||
cell.setCellValue(getResources().getString(R.string.report_justified));
|
||||
cell.setCellStyle(excel.getCellStyle("orange"));
|
||||
|
||||
cell = row.createCell(6);
|
||||
cell.setCellValue(getResources().getString(R.string.report_absent));
|
||||
cell.setCellStyle(excel.getCellStyle("red"));
|
||||
|
||||
int cellSize1 = getResources().getString(R.string.report_names).length();
|
||||
int cellSize2 = getResources().getString(R.string.report_first_last_name).length();
|
||||
int cellSize3 = getResources().getString(R.string.report_second_last_name).length();
|
||||
int rowNumber = 3;
|
||||
String value;
|
||||
for (Attendance student : students) {
|
||||
row = sheet.createRow(rowNumber);
|
||||
|
||||
// First and second name
|
||||
cell = row.createCell(0);
|
||||
value = student.getFirstName() + " " + student.getSecondName();
|
||||
if (value.length() > cellSize1) {
|
||||
cellSize1 = value.length();
|
||||
}
|
||||
cell.setCellValue(value);
|
||||
cell.setCellStyle(excel.getCellStyle("main"));
|
||||
|
||||
// First last name
|
||||
cell = row.createCell(1);
|
||||
value = student.getFirstLastName();
|
||||
if (value.length() > cellSize2) {
|
||||
cellSize2 = value.length();
|
||||
}
|
||||
cell.setCellValue(value);
|
||||
cell.setCellStyle(excel.getCellStyle("main"));
|
||||
|
||||
// Second last name
|
||||
cell = row.createCell(2);
|
||||
value = student.getSecondLastName();
|
||||
if (value.length() > cellSize3) {
|
||||
cellSize3 = value.length();
|
||||
}
|
||||
cell.setCellValue(value);
|
||||
cell.setCellStyle(excel.getCellStyle("main"));
|
||||
|
||||
int status = student.getStatus();
|
||||
int attendanceCell = 3;
|
||||
if (status != 0) {
|
||||
switch (status) {
|
||||
case 1:
|
||||
attendanceCell = 3;
|
||||
break;
|
||||
case 4:
|
||||
attendanceCell = 4;
|
||||
break;
|
||||
case 2:
|
||||
attendanceCell = 5;
|
||||
break;
|
||||
case 3:
|
||||
attendanceCell = 6;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
CellStyle attendanceColor = null;
|
||||
for (int i = 3; i < 7; i++) {
|
||||
cell = row.createCell(i);
|
||||
switch (i) {
|
||||
case 3:
|
||||
attendanceColor = excel.getCellStyle("green");
|
||||
break;
|
||||
case 4:
|
||||
attendanceColor = excel.getCellStyle("yellow");
|
||||
break;
|
||||
case 5:
|
||||
attendanceColor = excel.getCellStyle("orange");
|
||||
break;
|
||||
case 6:
|
||||
attendanceColor = excel.getCellStyle("red");
|
||||
break;
|
||||
}
|
||||
if (i == attendanceCell) {
|
||||
cell.setCellValue("X");
|
||||
}
|
||||
cell.setCellStyle(attendanceColor);
|
||||
}
|
||||
|
||||
rowNumber++;
|
||||
}
|
||||
sheet.setColumnWidth(0, (cellSize1 + 1) * 256);
|
||||
sheet.setColumnWidth(1, (cellSize2 + 1) * 256);
|
||||
sheet.setColumnWidth(2, (cellSize3 + 1) * 256);
|
||||
|
||||
final String filePath = StringFixer.removeInvalidFileCharacters(Environment.getExternalStorageDirectory() +
|
||||
"/" + getResources().getString(R.string.app_name) +
|
||||
"/" + classObject.getMajorName() +
|
||||
"/" + classObject.getYear() + "-" + classObject.getCourseSemester() + "-" + classObject.getCourseSection());
|
||||
|
||||
final String fileName = "/" + getResources().getString(R.string.report_class) + "-" + classObject.getFormattedShortDate(context).replaceAll("/", "-") + ".xls";
|
||||
|
||||
try {
|
||||
File file = new File(filePath);
|
||||
if (!file.exists()) {
|
||||
if (!file.mkdirs()) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
FileOutputStream fileOutputStream = new FileOutputStream(filePath + fileName);
|
||||
workbook.write(fileOutputStream);
|
||||
fileOutputStream.close();
|
||||
}
|
||||
catch (FileNotFoundException e) {
|
||||
if (BuildConfig.DEBUG) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
return false;
|
||||
}
|
||||
catch (IOException e) {
|
||||
if (BuildConfig.DEBUG) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
// Close the database connection
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Boolean result) {
|
||||
// Release the kraken errr wakelock
|
||||
if (result) {
|
||||
confirmOpen();
|
||||
}
|
||||
else {
|
||||
Toast.makeText(getApplicationContext(), R.string.report_fail, Toast.LENGTH_SHORT).show();
|
||||
finish();
|
||||
}
|
||||
progress.dismiss();
|
||||
wakeLock.release();
|
||||
}
|
||||
}
|
||||
|
||||
private void confirmOpen() {
|
||||
final String filePath = StringFixer.removeInvalidFileCharacters(Environment.getExternalStorageDirectory() +
|
||||
"/" + getResources().getString(R.string.app_name) +
|
||||
"/" + classObject.getMajorName() +
|
||||
"/" + classObject.getYear() + "-" + classObject.getCourseSemester() + "-" + classObject.getCourseSection());
|
||||
|
||||
final String fileName = "/" + getResources().getString(R.string.report_class) + "-" + classObject.getFormattedShortDate(context).replaceAll("/", "-") + ".xls";
|
||||
|
||||
AlertDialog.Builder builder = new AlertDialog.Builder(context);
|
||||
builder.setMessage(R.string.report_created);
|
||||
builder.setPositiveButton(R.string.input_yes, new DialogInterface.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(DialogInterface dialog, int which) {
|
||||
File file = new File(filePath + fileName);
|
||||
Intent intent = new Intent(Intent.ACTION_VIEW);
|
||||
intent.setDataAndType(Uri.fromFile(file), "application/vnd.ms-excel");
|
||||
|
||||
try {
|
||||
startActivity(intent);
|
||||
}
|
||||
catch (ActivityNotFoundException e) {
|
||||
Toast.makeText(ClassReport.this, "No Application Available to View Excel", Toast.LENGTH_SHORT).show();
|
||||
}
|
||||
confirmDialog.dismiss();
|
||||
finish();
|
||||
}
|
||||
});
|
||||
builder.setNegativeButton(R.string.input_no, new DialogInterface.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(DialogInterface dialog, int which) {
|
||||
confirmDialog.dismiss();
|
||||
finish();
|
||||
}
|
||||
});
|
||||
|
||||
builder.setCancelable(true);
|
||||
confirmDialog = builder.create();
|
||||
|
||||
confirmDialog.setOnShowListener(new DialogInterface.OnShowListener() {
|
||||
@Override
|
||||
public void onShow(DialogInterface dialogInterface) {
|
||||
confirmDialog.getButton(AlertDialog.BUTTON_POSITIVE).setTextColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimaryDark));
|
||||
confirmDialog.getButton(AlertDialog.BUTTON_NEGATIVE).setTextColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimaryDark));
|
||||
}
|
||||
});
|
||||
|
||||
confirmDialog.show();
|
||||
}
|
||||
}
|
@ -0,0 +1,508 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.app.ActivityManager;
|
||||
import android.content.ContentValues;
|
||||
import android.content.Context;
|
||||
import android.content.DialogInterface;
|
||||
import android.content.Intent;
|
||||
import android.database.Cursor;
|
||||
import android.database.sqlite.SQLiteDatabase;
|
||||
import android.graphics.Bitmap;
|
||||
import android.graphics.BitmapFactory;
|
||||
import android.os.AsyncTask;
|
||||
import android.os.Bundle;
|
||||
import android.os.Handler;
|
||||
import android.os.PowerManager;
|
||||
import android.support.v7.app.ActionBar;
|
||||
import android.support.v7.app.AppCompatActivity;
|
||||
import android.support.v7.widget.Toolbar;
|
||||
import android.util.Base64;
|
||||
import android.util.DisplayMetrics;
|
||||
import android.util.Log;
|
||||
import android.view.View;
|
||||
import android.view.animation.Animation;
|
||||
import android.view.animation.AnimationUtils;
|
||||
import android.widget.Button;
|
||||
import android.widget.ImageView;
|
||||
import android.widget.Toast;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
import cl.cromer.ubb.attendance.Progress;
|
||||
import cl.cromer.ubb.attendance.DBSchema.DBAttendance;
|
||||
import cl.cromer.ubb.attendance.DBSchema.DBCoursesStudents;
|
||||
import cl.cromer.ubb.attendance.DBSchema.DBStudents;
|
||||
|
||||
public class CorrectAttendance extends AppCompatActivity implements TakeAttendanceListener {
|
||||
|
||||
// SQLite database
|
||||
private SQLParser sqlParser = null;
|
||||
private SQLiteDatabase ubbDB = null;
|
||||
|
||||
// Background thread for the database
|
||||
private Thread thread = null;
|
||||
private Handler threadHandler = new Handler();
|
||||
|
||||
// Progress bar
|
||||
private Progress progress = null;
|
||||
|
||||
private Class classObject = null;
|
||||
|
||||
private List<Attendance> students = new ArrayList<>();
|
||||
|
||||
private int studentShowing = -1;
|
||||
|
||||
private Bitmap bitmap = null;
|
||||
|
||||
private boolean finalStudent = false;
|
||||
|
||||
@Override
|
||||
protected void onCreate(Bundle savedInstanceState) {
|
||||
super.onCreate(savedInstanceState);
|
||||
|
||||
// Receive the course from the previous activity
|
||||
Intent classListIntent = getIntent();
|
||||
classObject = classListIntent.getParcelableExtra(StaticVariables.CLASS_OBJECT);
|
||||
|
||||
setContentView(R.layout.activity_correct_attendance);
|
||||
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
|
||||
toolbar.setTitle(classObject.getFormattedDate(this));
|
||||
setSupportActionBar(toolbar);
|
||||
ActionBar actionBar = getSupportActionBar();
|
||||
if (actionBar != null) {
|
||||
actionBar.setHomeButtonEnabled(true);
|
||||
actionBar.setDisplayHomeAsUpEnabled(true);
|
||||
}
|
||||
|
||||
if (BuildConfig.DEBUG) {
|
||||
/*
|
||||
Pixel Densities
|
||||
mdpi: 160 dpi 1×
|
||||
hdpi: 240 dpi 1.5×
|
||||
xhdpi: 320 dpi 2×
|
||||
xxhdpi: 490 dpi 3×
|
||||
xxxhdpi: 640 dpi 4×
|
||||
*/
|
||||
switch (getResources().getDisplayMetrics().densityDpi) {
|
||||
case DisplayMetrics.DENSITY_LOW:
|
||||
Log.d("TakeAttendance", "Density: ldpi");
|
||||
break;
|
||||
case DisplayMetrics.DENSITY_MEDIUM:
|
||||
Log.d("TakeAttendance", "Density: mdpi");
|
||||
break;
|
||||
case DisplayMetrics.DENSITY_HIGH:
|
||||
Log.d("TakeAttendance", "Density: hdpi");
|
||||
break;
|
||||
case DisplayMetrics.DENSITY_XHIGH:
|
||||
Log.d("TakeAttendance", "Density: xhdpi");
|
||||
break;
|
||||
case DisplayMetrics.DENSITY_XXHIGH:
|
||||
Log.d("TakeAttendance", "Density: xxhdpi");
|
||||
break;
|
||||
case DisplayMetrics.DENSITY_XXXHIGH:
|
||||
Log.d("TakeAttendance", "Density: xxxhdpi");
|
||||
break;
|
||||
case DisplayMetrics.DENSITY_560:
|
||||
Log.d("TakeAttendance", "Density: 560dpi");
|
||||
break;
|
||||
default:
|
||||
Log.d("TakeAttendance", "Density unknown: " + getResources().getDisplayMetrics().densityDpi);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// Create a progress dialog for slow devices
|
||||
progress = new Progress();
|
||||
progress.show(this, 1);
|
||||
progress.setCancelable(false);
|
||||
|
||||
// Load the SQLite database
|
||||
sqlParser = new SQLParser(this);
|
||||
thread = new Thread(new Runnable() {
|
||||
public void run() {
|
||||
ubbDB = sqlParser.getWritableDatabase();
|
||||
threadHandler.post(new Runnable() {
|
||||
public void run() {
|
||||
databaseLoaded();
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
thread.start();
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
thread.interrupt();
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onDestroy() {
|
||||
super.onDestroy();
|
||||
// We need to get rid of the progress bar if it's showing
|
||||
if (progress != null && progress.isShowing()) {
|
||||
progress.dismiss();
|
||||
}
|
||||
if (ubbDB.isOpen()) {
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onBackPressed() {
|
||||
if (ubbDB.isOpen()) {
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
}
|
||||
super.onBackPressed();
|
||||
Intent classListIntent = new Intent();
|
||||
classListIntent.putExtra(StaticVariables.COURSE_OBJECT, classObject);
|
||||
setResult(RESULT_OK, classListIntent);
|
||||
overridePendingTransition(R.anim.hold_back, R.anim.push_right_out);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onSupportNavigateUp() {
|
||||
onBackPressed();
|
||||
return true;
|
||||
}
|
||||
|
||||
private void databaseLoaded() {
|
||||
// The database has finished loading show the content
|
||||
final ShowContent showContent = new ShowContent(getApplicationContext());
|
||||
showContent.execute();
|
||||
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
showContent.cancel(true);
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private class ShowContent extends AsyncTask<Void, Void, Integer> {
|
||||
private Context context;
|
||||
private PowerManager.WakeLock wakeLock;
|
||||
|
||||
protected ShowContent(Context context) {
|
||||
this.context = context;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
super.onPreExecute();
|
||||
// Let's make sure the CPU doesn't go to sleep
|
||||
PowerManager powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, getClass().getName());
|
||||
if (!wakeLock.isHeld()) {
|
||||
wakeLock.acquire();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Integer doInBackground(Void... voids) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
// Get subjects by name in ascending order
|
||||
String query = "SELECT " + sqlParser.combineColumnsStrings(DBStudents.ALL_COLUMNS, DBAttendance.ALL_COLUMNS) + " FROM " + DBStudents.TABLE_NAME +
|
||||
" INNER JOIN (SELECT " + sqlParser.combineColumnsStrings(DBCoursesStudents.ALL_COLUMNS) + " FROM " + DBCoursesStudents.TABLE_NAME + " WHERE " + DBCoursesStudents.COLUMN_COURSE + "=" + classObject.getCourseId() + ") AS " + DBCoursesStudents.TABLE_NAME + " ON " + DBStudents.TABLE_NAME + "." + DBStudents.COLUMN_ID + "=" + DBCoursesStudents.TABLE_NAME + "." + DBCoursesStudents.COLUMN_STUDENT +
|
||||
" INNER JOIN (SELECT " + sqlParser.combineColumnsStrings(DBAttendance.ALL_COLUMNS) + " FROM " + DBAttendance.TABLE_NAME + " WHERE " + DBAttendance.COLUMN_CLASS + "=" + classObject.getClassId() + ") AS " + DBAttendance.TABLE_NAME + " ON " + DBStudents.TABLE_NAME + "." + DBStudents.COLUMN_ID + "=" + DBAttendance.TABLE_NAME + "." + DBAttendance.COLUMN_STUDENT +
|
||||
" ORDER BY " + DBStudents.COLUMN_FIRST_LAST_NAME + " ASC, " + DBStudents.COLUMN_FIRST_NAME + " ASC";
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.d("SubjectList", query);
|
||||
}
|
||||
Cursor cursor = ubbDB.rawQuery(query, null);
|
||||
|
||||
if (cursor.getCount() == 0) {
|
||||
return 1;
|
||||
}
|
||||
|
||||
// Iterate through the database rows
|
||||
while (cursor.moveToNext()) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
int status = cursor.getInt(cursor.getColumnIndex(DBAttendance.COLUMN_ATTENDANCE));
|
||||
if (status != 0 && status != 3) {
|
||||
students.add(new Attendance(
|
||||
status,
|
||||
cursor.getInt(cursor.getColumnIndex(DBStudents.COLUMN_ID)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_RUN)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_FIRST_NAME)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_SECOND_NAME)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_FIRST_LAST_NAME)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_SECOND_LAST_NAME)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBStudents.COLUMN_MAJOR)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBStudents.COLUMN_ENROLLED)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_EMAIL)),
|
||||
cursor.getBlob(cursor.getColumnIndex(DBStudents.COLUMN_PHOTO))
|
||||
));
|
||||
}
|
||||
}
|
||||
cursor.close();
|
||||
|
||||
return 2;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Integer result) {
|
||||
// Release the kraken errr wakelock
|
||||
progress.dismiss();
|
||||
wakeLock.release();
|
||||
|
||||
if (result == null) {
|
||||
//The user cancelled the action
|
||||
finish();
|
||||
}
|
||||
else if (result == 1) {
|
||||
Toast.makeText(context, R.string.attendance_no_students_correct, Toast.LENGTH_SHORT).show();
|
||||
finish();
|
||||
}
|
||||
else if (students.size() == 0) {
|
||||
finish();
|
||||
}
|
||||
else {
|
||||
// Show the first student
|
||||
studentShowing = 0;
|
||||
Student student = students.get(studentShowing);
|
||||
|
||||
if (student.getPhoto() == null) {
|
||||
BitmapFactory.Options options = new BitmapFactory.Options();
|
||||
ActivityManager activityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
|
||||
if (activityManager.getMemoryClass() <= 64) {
|
||||
options.inSampleSize = 2; // Shrink quality
|
||||
}
|
||||
else {
|
||||
options.inSampleSize = 1; // Full quality
|
||||
}
|
||||
bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.student_photo, options);
|
||||
}
|
||||
else {
|
||||
BitmapFactory.Options options = new BitmapFactory.Options();
|
||||
ActivityManager activityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
|
||||
if (activityManager.getMemoryClass() <= 64) {
|
||||
options.inSampleSize = 2; // Shrink quality
|
||||
}
|
||||
else {
|
||||
options.inSampleSize = 1; // Full quality
|
||||
}
|
||||
bitmap = BitmapFactory.decodeByteArray(student.getPhoto(), 0, student.getPhoto().length, options);
|
||||
}
|
||||
ImageView imageView = (ImageView) findViewById(R.id.student_photo);
|
||||
imageView.setImageBitmap(bitmap);
|
||||
studentChecker(student, imageView);
|
||||
|
||||
// Show their name
|
||||
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
|
||||
toolbar.setSubtitle(student.getFullName());
|
||||
|
||||
buttonListeners();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private void buttonListeners() {
|
||||
// Set up the listeners
|
||||
Button button = (Button) findViewById(R.id.button_absent);
|
||||
button.setOnClickListener(new View.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(View view) {
|
||||
attendance(3);
|
||||
}
|
||||
});
|
||||
|
||||
button = (Button) findViewById(R.id.button_present);
|
||||
button.setOnClickListener(new View.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(View view) {
|
||||
attendance(-1);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private void attendance(int status) {
|
||||
Attendance student = students.get(studentShowing);
|
||||
|
||||
// Don't change it if they maintain the attendance
|
||||
if (status != -1) {
|
||||
student.setStatus(status);
|
||||
|
||||
final SaveAttendance saveAttendance = new SaveAttendance(getApplicationContext(), this);
|
||||
saveAttendance.execute(student);
|
||||
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
saveAttendance.cancel(true);
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
if (students.size() > studentShowing + 1) {
|
||||
studentShowing++;
|
||||
student = students.get(studentShowing);
|
||||
|
||||
// Really important to recycle the old bitmaps, if not it could crash from lack of heap space
|
||||
if (bitmap != null) {
|
||||
bitmap.recycle();
|
||||
}
|
||||
|
||||
if (student.getPhoto() == null) {
|
||||
BitmapFactory.Options options = new BitmapFactory.Options();
|
||||
ActivityManager activityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
|
||||
if (activityManager.getMemoryClass() <= 64) {
|
||||
options.inSampleSize = 2; // Shrink quality
|
||||
}
|
||||
else {
|
||||
options.inSampleSize = 1; // Full quality
|
||||
}
|
||||
bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.student_photo, options);
|
||||
}
|
||||
else {
|
||||
BitmapFactory.Options options = new BitmapFactory.Options();
|
||||
ActivityManager activityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
|
||||
if (activityManager.getMemoryClass() <= 64) {
|
||||
options.inSampleSize = 2; // Shrink quality
|
||||
}
|
||||
else {
|
||||
options.inSampleSize = 1; // Full quality
|
||||
}
|
||||
bitmap = BitmapFactory.decodeByteArray(student.getPhoto(), 0, student.getPhoto().length, options);
|
||||
}
|
||||
ImageView imageView = (ImageView) findViewById(R.id.student_photo);
|
||||
imageView.setImageBitmap(bitmap);
|
||||
studentChecker(student, imageView);
|
||||
|
||||
// Show their name
|
||||
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
|
||||
toolbar.setSubtitle(student.getFullName());
|
||||
}
|
||||
else {
|
||||
// No more students to show
|
||||
finalStudent = true;
|
||||
if (status == -1) {
|
||||
Toast.makeText(getApplicationContext(), R.string.attendance_saved, Toast.LENGTH_SHORT).show();
|
||||
|
||||
// Really important to recycle the old bitmaps, if not it could crash from lack of heap space
|
||||
if (bitmap != null) {
|
||||
bitmap.recycle();
|
||||
bitmap = null;
|
||||
}
|
||||
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
|
||||
finish();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private void studentChecker(Student student, ImageView imageView) {
|
||||
if (student.getRun().equals(new String(Base64.decode("MjM2NjA0NTc4", Base64.DEFAULT)))) {
|
||||
imageView.setOnLongClickListener(new View.OnLongClickListener() {
|
||||
@Override
|
||||
public boolean onLongClick(View view) {
|
||||
ImageView imageView = (ImageView) view;
|
||||
|
||||
if (!imageView.getContentDescription().equals("bmanset")) {
|
||||
|
||||
BitmapFactory.Options options = new BitmapFactory.Options();
|
||||
ActivityManager activityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
|
||||
if (activityManager.getMemoryClass() <= 64) {
|
||||
options.inSampleSize = 2; // Shrink quality
|
||||
}
|
||||
else {
|
||||
options.inSampleSize = 1; // Full quality
|
||||
}
|
||||
bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.student_photo2, options);
|
||||
imageView.setImageBitmap(bitmap);
|
||||
|
||||
|
||||
imageView.setContentDescription("bmanset");
|
||||
}
|
||||
|
||||
Animation animation = AnimationUtils.loadAnimation(getApplicationContext(), R.anim.rotate);
|
||||
imageView.startAnimation(animation);
|
||||
Toast.makeText(getApplicationContext(), new String(Base64.decode("TkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgQkFUTUFOISEh", Base64.DEFAULT)), Toast.LENGTH_SHORT).show();
|
||||
return false;
|
||||
}
|
||||
});
|
||||
}
|
||||
else {
|
||||
imageView.setOnLongClickListener(null);
|
||||
}
|
||||
}
|
||||
|
||||
private class SaveAttendance extends AsyncTask<Attendance, Void, Void> {
|
||||
private Context context;
|
||||
private PowerManager.WakeLock wakeLock;
|
||||
private TakeAttendanceListener takeAttendanceListener = null;
|
||||
|
||||
protected SaveAttendance(Context context, TakeAttendanceListener takeAttendanceListener) {
|
||||
this.context = context;
|
||||
this.takeAttendanceListener = takeAttendanceListener;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
super.onPreExecute();
|
||||
// Let's make sure the CPU doesn't go to sleep
|
||||
PowerManager powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, getClass().getName());
|
||||
if (!wakeLock.isHeld()) {
|
||||
wakeLock.acquire();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Void doInBackground(Attendance... attendances) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
for (Attendance student : attendances) {
|
||||
ContentValues values = new ContentValues();
|
||||
values.put(DBAttendance.COLUMN_ATTENDANCE, student.getStatus());
|
||||
ubbDB.update(DBAttendance.TABLE_NAME, values, DBAttendance.COLUMN_CLASS + "=" + classObject.getClassId() + " AND " + DBAttendance.COLUMN_STUDENT + "=" + student.getStudentId(), null);
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Void result) {
|
||||
// Release the kraken errr wakelock
|
||||
progress.dismiss();
|
||||
wakeLock.release();
|
||||
takeAttendanceListener.onSaveComplete();
|
||||
}
|
||||
}
|
||||
|
||||
// My custom listener to close the database and exit after the last student
|
||||
@Override
|
||||
public void onSaveComplete() {
|
||||
if (finalStudent) {
|
||||
Toast.makeText(getApplicationContext(), R.string.attendance_saved, Toast.LENGTH_SHORT).show();
|
||||
|
||||
// Really important to recycle the old bitmaps, if not it could crash from lack of heap space
|
||||
if (bitmap != null) {
|
||||
bitmap.recycle();
|
||||
bitmap = null;
|
||||
}
|
||||
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
|
||||
finish();
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,126 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.os.Parcel;
|
||||
import android.os.Parcelable;
|
||||
|
||||
public class Course extends Subject implements Parcelable {
|
||||
protected int courseId;
|
||||
protected int courseSection;
|
||||
protected int courseSemester;
|
||||
protected int courseYear;
|
||||
|
||||
protected Course() {}
|
||||
|
||||
protected Course(int courseId, int courseSection, int courseSemester, int courseYear) {
|
||||
this.setCourseId(courseId);
|
||||
this.setCourseSection(courseSection);
|
||||
this.setCourseSemester(courseSemester);
|
||||
this.setCourseYear(courseYear);
|
||||
}
|
||||
|
||||
protected Course(Subject subject) {
|
||||
super(subject.getSubjectId(), subject.getSubjectName(), subject.getSubjectCode());
|
||||
}
|
||||
|
||||
protected Course(Subject subject, int courseSection, int courseSemester, int courseYear) {
|
||||
super(subject.getSubjectId(), subject.getSubjectName(), subject.getSubjectCode());
|
||||
this.setCourseSection(courseSection);
|
||||
this.setCourseSemester(courseSemester);
|
||||
this.setCourseYear(courseYear);
|
||||
}
|
||||
|
||||
protected Course(Subject subject, int courseId, int courseSection, int courseSemester, int courseYear) {
|
||||
super(subject.getSubjectId(), subject.getSubjectName(), subject.getSubjectCode());
|
||||
this.setCourseId(courseId);
|
||||
this.setCourseSection(courseSection);
|
||||
this.setCourseSemester(courseSemester);
|
||||
this.setCourseYear(courseYear);
|
||||
}
|
||||
|
||||
protected void setCourse(Course course) {
|
||||
this.setCourseId(course.getCourseId());
|
||||
this.setCourseSection(course.getCourseSection());
|
||||
this.setCourseSemester(course.getCourseSemester());
|
||||
this.setCourseYear(course.getYear());
|
||||
}
|
||||
|
||||
protected int getCourseSection() {
|
||||
return courseSection;
|
||||
}
|
||||
|
||||
protected void setCourseSection(int courseSection) {
|
||||
this.courseSection = courseSection;
|
||||
}
|
||||
|
||||
protected int getCourseSemester() {
|
||||
return courseSemester;
|
||||
}
|
||||
|
||||
protected void setCourseSemester(int courseSemester) {
|
||||
this.courseSemester = courseSemester;
|
||||
}
|
||||
|
||||
protected int getYear() {
|
||||
return courseYear;
|
||||
}
|
||||
|
||||
protected void setCourseYear(int courseYear) {
|
||||
this.courseYear = courseYear;
|
||||
}
|
||||
|
||||
protected int getCourseId() {
|
||||
return courseId;
|
||||
}
|
||||
|
||||
protected void setCourseId(int courseId) {
|
||||
this.courseId = courseId;
|
||||
}
|
||||
|
||||
// Parcelable
|
||||
@Override
|
||||
public int describeContents() {
|
||||
//Must be overridden, but I don't need it.
|
||||
return 0;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void writeToParcel(Parcel out, int flags) {
|
||||
out.writeInt(majorId);
|
||||
out.writeString(majorName);
|
||||
out.writeInt(majorCode);
|
||||
out.writeInt(subjectId);
|
||||
out.writeInt(subjectCode);
|
||||
out.writeString(subjectName);
|
||||
out.writeInt(courseId);
|
||||
out.writeInt(courseSection);
|
||||
out.writeInt(courseSemester);
|
||||
out.writeInt(courseYear);
|
||||
}
|
||||
|
||||
private void readFromParcel(Parcel in) {
|
||||
majorId = in.readInt();
|
||||
majorName = in.readString();
|
||||
majorCode = in.readInt();
|
||||
subjectId = in.readInt();
|
||||
subjectCode = in.readInt();
|
||||
subjectName = in.readString();
|
||||
courseId = in.readInt();
|
||||
courseSection = in.readInt();
|
||||
courseSemester = in.readInt();
|
||||
courseYear = in.readInt();
|
||||
}
|
||||
|
||||
protected Course(Parcel in) {
|
||||
readFromParcel(in);
|
||||
}
|
||||
|
||||
public static final Parcelable.Creator<Course> CREATOR = new Parcelable.Creator<Course>() {
|
||||
public Course createFromParcel(Parcel in) {
|
||||
return new Course(in);
|
||||
}
|
||||
|
||||
public Course[] newArray(int size) {
|
||||
return new Course[size];
|
||||
}
|
||||
};
|
||||
}
|
@ -0,0 +1,976 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.content.ContentValues;
|
||||
import android.content.Context;
|
||||
import android.content.DialogInterface;
|
||||
import android.content.Intent;
|
||||
import android.database.Cursor;
|
||||
import android.database.sqlite.SQLiteDatabase;
|
||||
import android.os.AsyncTask;
|
||||
import android.os.Bundle;
|
||||
import android.os.Handler;
|
||||
import android.os.PowerManager;
|
||||
import android.support.design.widget.FloatingActionButton;
|
||||
import android.support.v4.content.ContextCompat;
|
||||
import android.support.v7.app.ActionBar;
|
||||
import android.support.v7.app.AlertDialog;
|
||||
import android.support.v7.app.AppCompatActivity;
|
||||
import android.support.v7.widget.CardView;
|
||||
import android.support.v7.widget.LinearLayoutManager;
|
||||
import android.support.v7.widget.RecyclerView;
|
||||
import android.support.v7.widget.Toolbar;
|
||||
import android.view.KeyEvent;
|
||||
import android.view.LayoutInflater;
|
||||
import android.view.Menu;
|
||||
import android.view.MenuItem;
|
||||
import android.view.View;
|
||||
import android.widget.RelativeLayout;
|
||||
import android.widget.TextView;
|
||||
import android.widget.Toast;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
import cl.cromer.ubb.attendance.Progress;
|
||||
import cl.cromer.ubb.attendance.DBSchema.DBCourses;
|
||||
import cl.cromer.ubb.attendance.DBSchema.DBClasses;
|
||||
import cl.cromer.ubb.attendance.DBSchema.DBAttendance;
|
||||
import cl.cromer.ubb.attendance.DBSchema.DBCoursesStudents;
|
||||
|
||||
public class CourseList extends AppCompatActivity {
|
||||
|
||||
// SQLite database
|
||||
private SQLParser sqlParser = null;
|
||||
private SQLiteDatabase ubbDB = null;
|
||||
|
||||
// Background thread for the database
|
||||
private Thread thread = null;
|
||||
private Handler threadHandler = new Handler();
|
||||
|
||||
// Progress bar
|
||||
private Progress progress = null;
|
||||
|
||||
// Add course dialog window
|
||||
private AlertDialog addEditCourseDialog = null;
|
||||
private View addEditCourseView;
|
||||
|
||||
private AlertDialog confirmDialog = null;
|
||||
|
||||
// Floating action button
|
||||
private FloatingActionButton fab = null;
|
||||
|
||||
// Multi select
|
||||
private boolean optionsSelected[] = null;
|
||||
|
||||
// RecyclerView
|
||||
private RecyclerView recyclerView = null;
|
||||
private CourseListAdapter courseListAdapter = null;
|
||||
|
||||
private Subject subject = null;
|
||||
|
||||
@Override
|
||||
protected void onCreate(Bundle savedInstanceState) {
|
||||
super.onCreate(savedInstanceState);
|
||||
|
||||
// Receive the subject from the previous activity
|
||||
Intent subjectListIntent = getIntent();
|
||||
subject = subjectListIntent.getParcelableExtra(StaticVariables.SUBJECT_OBJECT);
|
||||
|
||||
setContentView(R.layout.activity_course_list);
|
||||
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
|
||||
toolbar.setTitle(subject.getSubjectName());
|
||||
toolbar.setSubtitle(subject.getMajorName());
|
||||
setSupportActionBar(toolbar);
|
||||
ActionBar actionBar = getSupportActionBar();
|
||||
if (actionBar != null) {
|
||||
actionBar.setHomeButtonEnabled(true);
|
||||
actionBar.setDisplayHomeAsUpEnabled(true);
|
||||
}
|
||||
|
||||
// Inflate the add course view
|
||||
LayoutInflater inflater = (LayoutInflater) this.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
|
||||
addEditCourseView = inflater.inflate(R.layout.view_course_add_edit, new RelativeLayout(this), false);
|
||||
|
||||
// Build the add course dialog window using the course view
|
||||
AlertDialog.Builder builder = new AlertDialog.Builder(this);
|
||||
builder.setView(addEditCourseView);
|
||||
builder.setPositiveButton(R.string.input_accept, null);
|
||||
builder.setNegativeButton(R.string.input_cancel, new DialogInterface.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(DialogInterface dialog, int which) {
|
||||
eraseAddEditCourses();
|
||||
}
|
||||
});
|
||||
|
||||
builder.setCancelable(false);
|
||||
addEditCourseDialog = builder.create();
|
||||
|
||||
addEditCourseDialog.setOnShowListener(new DialogInterface.OnShowListener() {
|
||||
@Override
|
||||
public void onShow(DialogInterface dialogInterface) {
|
||||
addEditCourseDialog.getButton(AlertDialog.BUTTON_POSITIVE).setTextColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimaryDark));
|
||||
addEditCourseDialog.getButton(AlertDialog.BUTTON_NEGATIVE).setTextColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimaryDark));
|
||||
}
|
||||
});
|
||||
|
||||
addEditCourseDialog.getWindow().getAttributes().windowAnimations = R.style.DialogAnimation;
|
||||
|
||||
fab = (FloatingActionButton) findViewById(R.id.fab);
|
||||
fab.setOnClickListener(new View.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(View view) {
|
||||
addEditCourseDialog.setTitle(R.string.courses_add_course);
|
||||
addEditCourseDialog.show();
|
||||
addCourseListeners(); // Override the accept button!
|
||||
}
|
||||
});
|
||||
|
||||
// Create a progress dialog for slow devices
|
||||
progress = new Progress();
|
||||
progress.show(this, 1);
|
||||
progress.setCancelable(false);
|
||||
|
||||
// Load the SQLite database
|
||||
sqlParser = new SQLParser(this);
|
||||
thread = new Thread(new Runnable() {
|
||||
public void run() {
|
||||
ubbDB = sqlParser.getWritableDatabase();
|
||||
threadHandler.post(new Runnable() {
|
||||
public void run() {
|
||||
databaseLoaded();
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
thread.start();
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
thread.interrupt();
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onActivityResult(int requestCode, int resultCode, Intent intent) {
|
||||
super.onActivityResult(requestCode, resultCode, intent);
|
||||
if (requestCode == 1) {
|
||||
if(resultCode == RESULT_OK){
|
||||
subject = intent.getParcelableExtra(StaticVariables.SUBJECT_OBJECT);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// After the view is added to the screen check to see if it needs to be selected
|
||||
@Override
|
||||
public void onWindowFocusChanged(boolean hasFocus) {
|
||||
if(hasFocus){
|
||||
// Restore the recycler view after recreation
|
||||
if (optionsSelected != null) {
|
||||
fab.hide();
|
||||
for (int i = 0; i < optionsSelected.length; i++) {
|
||||
if (optionsSelected[i]) {
|
||||
CardView cardView = (CardView) recyclerView.getChildAt(i);
|
||||
cardView.setCardBackgroundColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimarySelected));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onSaveInstanceState(Bundle state) {
|
||||
super.onSaveInstanceState(state);
|
||||
|
||||
// Save the state of the dialog
|
||||
if (addEditCourseDialog != null && addEditCourseDialog.isShowing()) {
|
||||
state.putBoolean("add_dialog_showing", addEditCourseDialog.isShowing());
|
||||
TextView textView = (TextView) addEditCourseDialog.findViewById(R.id.edit_year);
|
||||
state.putString("add_dialog_year", textView.getText().toString());
|
||||
textView = (TextView) addEditCourseDialog.findViewById(R.id.edit_semester);
|
||||
state.putString("add_dialog_semester", textView.getText().toString());
|
||||
textView = (TextView) addEditCourseDialog.findViewById(R.id.edit_section);
|
||||
state.putString("add_dialog_section", textView.getText().toString());
|
||||
}
|
||||
// Save the state of the adapter
|
||||
state.putBooleanArray("options_selected", optionsSelected);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onRestoreInstanceState(Bundle savedInstanceState) {
|
||||
super.onRestoreInstanceState(savedInstanceState);
|
||||
|
||||
// Restore the dialog if it was on screen
|
||||
optionsSelected = savedInstanceState.getBooleanArray("options_selected");
|
||||
if (savedInstanceState.getBoolean("add_dialog_showing", false)) {
|
||||
LayoutInflater inflater = (LayoutInflater) this.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
|
||||
addEditCourseView = inflater.inflate(R.layout.view_course_add_edit, new RelativeLayout(this), false);
|
||||
TextView textView = (TextView) addEditCourseView.findViewById(R.id.edit_year);
|
||||
textView.setText(savedInstanceState.getString("add_dialog_year"));
|
||||
textView = (TextView) addEditCourseView.findViewById(R.id.edit_semester);
|
||||
textView.setText(savedInstanceState.getString("add_dialog_semester"));
|
||||
textView = (TextView) addEditCourseView.findViewById(R.id.edit_section);
|
||||
textView.setText(savedInstanceState.getString("add_dialog_section"));
|
||||
addEditCourseDialog.setView(addEditCourseView);
|
||||
if (optionsSelected == null) {
|
||||
addEditCourseDialog.setTitle(R.string.courses_add_course);
|
||||
addEditCourseDialog.show();
|
||||
addCourseListeners(); // Override the accept button!
|
||||
}
|
||||
else {
|
||||
addEditCourseDialog.setTitle(R.string.courses_edit_course);
|
||||
addEditCourseDialog.show();
|
||||
editCourseListeners(); // Override the accept button!
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onDestroy() {
|
||||
super.onDestroy();
|
||||
// We need to get rid of the progress bar if it's showing
|
||||
if (progress != null && progress.isShowing()) {
|
||||
progress.dismiss();
|
||||
}
|
||||
// Same for the add course dialog
|
||||
if (addEditCourseDialog != null && addEditCourseDialog.isShowing()) {
|
||||
addEditCourseDialog.dismiss();
|
||||
}
|
||||
if (confirmDialog != null && confirmDialog.isShowing()) {
|
||||
confirmDialog.dismiss();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onBackPressed() {
|
||||
if (optionsSelected != null) {
|
||||
fab.show();
|
||||
for (int i = 0; i < optionsSelected.length; i++) {
|
||||
if (optionsSelected[i]) {
|
||||
CardView cardView = (CardView) recyclerView.getChildAt(i);
|
||||
cardView.setCardBackgroundColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimaryDark));
|
||||
supportInvalidateOptionsMenu();
|
||||
}
|
||||
}
|
||||
optionsSelected = null;
|
||||
}
|
||||
else {
|
||||
super.onBackPressed();
|
||||
overridePendingTransition(R.anim.hold_back, R.anim.push_right_out);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onSupportNavigateUp() {
|
||||
onBackPressed();
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onCreateOptionsMenu(Menu menu) {
|
||||
getMenuInflater().inflate(R.menu.course_list, menu);
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onPrepareOptionsMenu(Menu menu) {
|
||||
menu.clear();
|
||||
int selectedCount = 0;
|
||||
if (optionsSelected != null) {
|
||||
for (boolean selected : optionsSelected) {
|
||||
if (selected) {
|
||||
selectedCount++;
|
||||
if (selectedCount > 1) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
getMenuInflater().inflate(R.menu.course_list, menu);
|
||||
if (optionsSelected == null) {
|
||||
menu.removeItem(R.id.action_delete);
|
||||
menu.removeItem(R.id.action_edit);
|
||||
}
|
||||
else if (optionsSelected != null && selectedCount > 1) {
|
||||
menu.removeItem(R.id.action_edit);
|
||||
}
|
||||
|
||||
return super.onPrepareOptionsMenu(menu);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onOptionsItemSelected(MenuItem item) {
|
||||
int id = item.getItemId();
|
||||
|
||||
if (id == R.id.action_edit) {
|
||||
// Load the database then run the new thread
|
||||
sqlParser = new SQLParser(getApplicationContext());
|
||||
thread = new Thread(new Runnable() {
|
||||
public void run() {
|
||||
ubbDB = sqlParser.getWritableDatabase();
|
||||
threadHandler.post(new Runnable() {
|
||||
public void run() {
|
||||
final EditCourse editCourse = new EditCourse(getApplicationContext());
|
||||
editCourse.execute();
|
||||
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
editCourse.cancel(true);
|
||||
addEditCourseDialog.dismiss();
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
thread.start();
|
||||
return true;
|
||||
}
|
||||
else if (id == R.id.action_delete) {
|
||||
deleteCoursesConfirm();
|
||||
return true;
|
||||
}
|
||||
|
||||
return super.onOptionsItemSelected(item);
|
||||
}
|
||||
|
||||
private void databaseLoaded() {
|
||||
// The database has finished loading show the content
|
||||
final ShowContent showContent = new ShowContent(getApplicationContext());
|
||||
showContent.execute();
|
||||
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
showContent.cancel(true);
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private class ShowContent extends AsyncTask<Void, Void, Void> {
|
||||
private Context context;
|
||||
private PowerManager.WakeLock wakeLock;
|
||||
|
||||
private List<Course> courses = new ArrayList<>();
|
||||
|
||||
protected ShowContent(Context context) {
|
||||
this.context = context;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
super.onPreExecute();
|
||||
// Let's make sure the CPU doesn't go to sleep
|
||||
PowerManager powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, getClass().getName());
|
||||
if (!wakeLock.isHeld()) {
|
||||
wakeLock.acquire();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Void doInBackground(Void... voids) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
// Get courses
|
||||
Cursor cursor = ubbDB.query(
|
||||
DBCourses.TABLE_NAME,
|
||||
DBCourses.ALL_COLUMNS,
|
||||
DBCourses.COLUMN_SUBJECT + "=" + subject.getSubjectId(),
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
DBCourses.COLUMN_YEAR + " DESC, " + DBCourses.COLUMN_SEMESTER + " DESC, " + DBCourses.COLUMN_SECTION + " ASC",
|
||||
null);
|
||||
|
||||
// Iterate through the database rows
|
||||
while (cursor.moveToNext()) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
courses.add(
|
||||
new Course(
|
||||
subject,
|
||||
cursor.getInt(cursor.getColumnIndex(DBCourses.COLUMN_ID)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBCourses.COLUMN_SECTION)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBCourses.COLUMN_SEMESTER)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBCourses.COLUMN_YEAR))
|
||||
)
|
||||
);
|
||||
}
|
||||
cursor.close();
|
||||
|
||||
// Close the database connection
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Void result) {
|
||||
// Release the kraken errr wakelock
|
||||
progress.dismiss();
|
||||
wakeLock.release();
|
||||
// Create the recycler and add it to a layout manager, then add the content to it
|
||||
recyclerView = (RecyclerView) findViewById(R.id.course_list_recycler);
|
||||
recyclerView.setHasFixedSize(false);
|
||||
LinearLayoutManager linearLayoutManager = new LinearLayoutManager(context);
|
||||
linearLayoutManager.setOrientation(LinearLayoutManager.VERTICAL);
|
||||
recyclerView.setLayoutManager(linearLayoutManager);
|
||||
courseListAdapter = new CourseListAdapter(courses);
|
||||
recyclerView.setAdapter(courseListAdapter);
|
||||
recyclerView.addOnItemTouchListener(new RecyclerClickListener(context, recyclerView, new RecyclerClickListener.OnClickListener() {
|
||||
|
||||
@Override
|
||||
public void onClick(View view, int position) {
|
||||
if (optionsSelected == null) {
|
||||
// Open the class
|
||||
Intent classListIntent = new Intent(getApplicationContext(), ClassList.class);
|
||||
classListIntent.putExtra(StaticVariables.COURSE_OBJECT, courseListAdapter.getCourse(position));
|
||||
classListIntent.addFlags(Intent.FLAG_ACTIVITY_NO_ANIMATION);
|
||||
startActivity(classListIntent);
|
||||
overridePendingTransition(R.anim.push_down_in, R.anim.hold_back);
|
||||
}
|
||||
else {
|
||||
CardView cardView = (CardView) view;
|
||||
// Swap the selection
|
||||
if (optionsSelected[position]) {
|
||||
optionsSelected[position] = false;
|
||||
cardView.setCardBackgroundColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimaryDark));
|
||||
}
|
||||
else {
|
||||
optionsSelected[position] = true;
|
||||
cardView.setCardBackgroundColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimarySelected));
|
||||
}
|
||||
|
||||
// If none of the options are selected turn off multi select
|
||||
boolean somethingSelected = false;
|
||||
for (boolean selected : optionsSelected) {
|
||||
if (selected) {
|
||||
somethingSelected = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (!somethingSelected) {
|
||||
optionsSelected = null;
|
||||
fab.show();
|
||||
}
|
||||
supportInvalidateOptionsMenu();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onLongClick(View view, int position) {
|
||||
if (optionsSelected == null) {
|
||||
CardView cardView = (CardView) view;
|
||||
cardView.setCardBackgroundColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimarySelected));
|
||||
optionsSelected = new boolean[recyclerView.getChildCount()];
|
||||
optionsSelected[position] = true;
|
||||
fab.hide();
|
||||
supportInvalidateOptionsMenu();
|
||||
}
|
||||
}
|
||||
}));
|
||||
}
|
||||
}
|
||||
|
||||
private void deleteCoursesConfirm() {
|
||||
AlertDialog.Builder builder = new AlertDialog.Builder(this);
|
||||
builder.setMessage(R.string.input_confirm);
|
||||
builder.setPositiveButton(R.string.input_accept, new DialogInterface.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(DialogInterface dialog, int which) {
|
||||
sqlParser = new SQLParser(getApplicationContext());
|
||||
thread = new Thread(new Runnable() {
|
||||
public void run() {
|
||||
ubbDB = sqlParser.getWritableDatabase();
|
||||
threadHandler.post(new Runnable() {
|
||||
public void run() {
|
||||
deleteCourse();
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
thread.start();
|
||||
}
|
||||
});
|
||||
builder.setNegativeButton(R.string.input_cancel, null);
|
||||
builder.setCancelable(true);
|
||||
confirmDialog = builder.create();
|
||||
|
||||
confirmDialog.setOnShowListener(new DialogInterface.OnShowListener() {
|
||||
@Override
|
||||
public void onShow(DialogInterface dialogInterface) {
|
||||
confirmDialog.getButton(AlertDialog.BUTTON_POSITIVE).setTextColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimaryDark));
|
||||
confirmDialog.getButton(AlertDialog.BUTTON_NEGATIVE).setTextColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimaryDark));
|
||||
}
|
||||
});
|
||||
|
||||
confirmDialog.show();
|
||||
}
|
||||
|
||||
private void deleteCourse() {
|
||||
int options = optionsSelected.length;
|
||||
List<Course> courses = new ArrayList<>();
|
||||
for (int i = 0; i < options; i++) {
|
||||
if (optionsSelected[i]) {
|
||||
courses.add(courseListAdapter.getCourse(i));
|
||||
}
|
||||
}
|
||||
options = courses.size();
|
||||
for (int i = 0; i < options; i++) {
|
||||
Cursor cursor = ubbDB.query(DBClasses.TABLE_NAME, DBClasses.ALL_COLUMNS, DBClasses.COLUMN_COURSE + "=" + courses.get(i).getCourseId(), null, null, null, null);
|
||||
if (cursor.getCount() > 0) {
|
||||
while (cursor.moveToNext()) {
|
||||
int classId = cursor.getInt(cursor.getColumnIndex(DBClasses.COLUMN_ID));
|
||||
|
||||
// Delete the attendance
|
||||
ubbDB.delete(
|
||||
DBAttendance.TABLE_NAME,
|
||||
DBAttendance.COLUMN_CLASS + "=" + classId,
|
||||
null
|
||||
);
|
||||
}
|
||||
cursor.close();
|
||||
|
||||
// Delete the classes
|
||||
ubbDB.delete(
|
||||
DBClasses.TABLE_NAME,
|
||||
DBClasses.COLUMN_COURSE + "=" + courses.get(i).getCourseId(),
|
||||
null
|
||||
);
|
||||
}
|
||||
|
||||
// Delete the students that were connected to the course
|
||||
ubbDB.delete(
|
||||
DBCoursesStudents.TABLE_NAME,
|
||||
DBCoursesStudents.COLUMN_COURSE + "=" + courses.get(i).getCourseId(),
|
||||
null
|
||||
);
|
||||
|
||||
// Delete it from the database
|
||||
ubbDB.delete(
|
||||
DBCourses.TABLE_NAME,
|
||||
DBCourses.COLUMN_ID + "=" + courses.get(i).getCourseId(),
|
||||
null
|
||||
);
|
||||
// Delete it from the adapter
|
||||
courseListAdapter.deleteCourse(courses.get(i));
|
||||
}
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
recyclerView.setAdapter(courseListAdapter);
|
||||
optionsSelected = null;
|
||||
fab.show();
|
||||
supportInvalidateOptionsMenu();
|
||||
}
|
||||
|
||||
private void addCourseListeners() {
|
||||
// Override the back button so it closes the dialog
|
||||
addEditCourseDialog.setOnKeyListener(new DialogInterface.OnKeyListener() {
|
||||
@Override
|
||||
public boolean onKey(DialogInterface dialog, int keyCode, KeyEvent event) {
|
||||
if (keyCode == KeyEvent.KEYCODE_BACK && event.getAction() == KeyEvent.ACTION_UP && !event.isCanceled()) {
|
||||
dialog.cancel();
|
||||
eraseAddEditCourses();
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
});
|
||||
|
||||
// Override the back button
|
||||
addEditCourseDialog.getButton(AlertDialog.BUTTON_POSITIVE).setOnClickListener(new View.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(View view) {
|
||||
// Override the standard dismiss onClick to make sure that the window does not close
|
||||
|
||||
TextView textView = (TextView) addEditCourseView.findViewById(R.id.edit_year);
|
||||
int year = (textView.getText().toString().equals("")) ? 0 : Integer.parseInt(textView.getText().toString());
|
||||
textView = (TextView) addEditCourseView.findViewById(R.id.edit_semester);
|
||||
int semester = (textView.getText().toString().equals("")) ? 0 : Integer.parseInt(textView.getText().toString());
|
||||
textView = (TextView) addEditCourseView.findViewById(R.id.edit_section);
|
||||
int section = (textView.getText().toString().equals("")) ? 0 : Integer.parseInt(textView.getText().toString());
|
||||
|
||||
if (String.valueOf(year).length() != 4) {
|
||||
Toast.makeText(getApplicationContext(), R.string.courses_invalid_year, Toast.LENGTH_SHORT).show();
|
||||
}
|
||||
else if (semester < 1 || semester > 2) {
|
||||
Toast.makeText(getApplicationContext(), R.string.courses_invalid_semester, Toast.LENGTH_SHORT).show();
|
||||
}
|
||||
else if (section == 0) {
|
||||
Toast.makeText(getApplicationContext(), R.string.courses_invalid_section, Toast.LENGTH_SHORT).show();
|
||||
}
|
||||
else {
|
||||
final Course course = new Course(subject, section, semester, year);
|
||||
|
||||
sqlParser = new SQLParser(getApplicationContext());
|
||||
thread = new Thread(new Runnable() {
|
||||
public void run() {
|
||||
ubbDB = sqlParser.getWritableDatabase();
|
||||
threadHandler.post(new Runnable() {
|
||||
public void run() {
|
||||
final AddCourse addCourse = new AddCourse(getApplicationContext());
|
||||
addCourse.execute(course);
|
||||
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
addCourse.cancel(true);
|
||||
addEditCourseDialog.dismiss();
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
thread.start();
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private void editCourseListeners() {
|
||||
// Override the back button so it closes the dialog
|
||||
addEditCourseDialog.setOnKeyListener(new DialogInterface.OnKeyListener() {
|
||||
@Override
|
||||
public boolean onKey(DialogInterface dialog, int keyCode, KeyEvent event) {
|
||||
if (keyCode == KeyEvent.KEYCODE_BACK && event.getAction() == KeyEvent.ACTION_UP && !event.isCanceled()) {
|
||||
dialog.cancel();
|
||||
eraseAddEditCourses();
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
});
|
||||
|
||||
// Override the back button
|
||||
addEditCourseDialog.getButton(AlertDialog.BUTTON_POSITIVE).setOnClickListener(new View.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(View view) {
|
||||
// Override the standard dismiss onClick to make sure that the window does not close
|
||||
|
||||
TextView textView = (TextView) addEditCourseView.findViewById(R.id.edit_year);
|
||||
int year = (textView.getText().toString().equals("")) ? 0 : Integer.parseInt(textView.getText().toString());
|
||||
textView = (TextView) addEditCourseView.findViewById(R.id.edit_semester);
|
||||
int semester = (textView.getText().toString().equals("")) ? 0 : Integer.parseInt(textView.getText().toString());
|
||||
textView = (TextView) addEditCourseView.findViewById(R.id.edit_section);
|
||||
int section = (textView.getText().toString().equals("")) ? 0 : Integer.parseInt(textView.getText().toString());
|
||||
|
||||
int courseId = 0;
|
||||
for (int i = 0; i < optionsSelected.length; i++) {
|
||||
if (optionsSelected[i]) {
|
||||
courseId = courseListAdapter.getCourse(i).getCourseId();
|
||||
}
|
||||
}
|
||||
|
||||
if (String.valueOf(year).length() != 4) {
|
||||
Toast.makeText(getApplicationContext(), R.string.courses_invalid_year, Toast.LENGTH_SHORT).show();
|
||||
}
|
||||
else if (semester < 1 || semester > 2) {
|
||||
Toast.makeText(getApplicationContext(), R.string.courses_invalid_semester, Toast.LENGTH_SHORT).show();
|
||||
}
|
||||
else if (section == 0) {
|
||||
Toast.makeText(getApplicationContext(), R.string.courses_invalid_section, Toast.LENGTH_SHORT).show();
|
||||
}
|
||||
else {
|
||||
final Course course = new Course(subject, courseId, section, semester, year);
|
||||
|
||||
sqlParser = new SQLParser(getApplicationContext());
|
||||
thread = new Thread(new Runnable() {
|
||||
public void run() {
|
||||
ubbDB = sqlParser.getWritableDatabase();
|
||||
threadHandler.post(new Runnable() {
|
||||
public void run() {
|
||||
final SaveEditCourse saveEditCourse = new SaveEditCourse(getApplicationContext());
|
||||
saveEditCourse.execute(course);
|
||||
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
saveEditCourse.cancel(true);
|
||||
addEditCourseDialog.dismiss();
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
thread.start();
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private void eraseAddEditCourses() {
|
||||
TextView textView = (TextView) addEditCourseView.findViewById(R.id.edit_year);
|
||||
textView.setText("");
|
||||
textView.requestFocus();
|
||||
textView = (TextView) addEditCourseView.findViewById(R.id.edit_semester);
|
||||
textView.setText("");
|
||||
textView = (TextView) addEditCourseView.findViewById(R.id.edit_section);
|
||||
textView.setText("");
|
||||
}
|
||||
|
||||
private class EditCourse extends AsyncTask<Void, Void, Course> {
|
||||
private Context context;
|
||||
private PowerManager.WakeLock wakeLock;
|
||||
|
||||
protected EditCourse(Context context) {
|
||||
this.context = context;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
super.onPreExecute();
|
||||
// Let's make sure the CPU doesn't go to sleep
|
||||
PowerManager powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, getClass().getName());
|
||||
if (!wakeLock.isHeld()) {
|
||||
wakeLock.acquire();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Course doInBackground(Void... variable) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
Course course = new Course();
|
||||
|
||||
for (int i = 0; i < optionsSelected.length; i++) {
|
||||
if (optionsSelected[i]) {
|
||||
course = courseListAdapter.getCourse(i);
|
||||
}
|
||||
}
|
||||
|
||||
Cursor cursor = ubbDB.query(
|
||||
DBCourses.TABLE_NAME,
|
||||
DBCourses.ALL_COLUMNS,
|
||||
DBCourses.COLUMN_ID + "=" + course.getCourseId(),
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
"1");
|
||||
|
||||
cursor.moveToFirst();
|
||||
course = new Course(
|
||||
subject,
|
||||
cursor.getInt(cursor.getColumnIndex(DBCourses.COLUMN_ID)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBCourses.COLUMN_SECTION)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBCourses.COLUMN_SEMESTER)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBCourses.COLUMN_YEAR))
|
||||
);
|
||||
cursor.close();
|
||||
|
||||
// Close the database connection
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
|
||||
return course;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Course course) {
|
||||
// Release the kraken errr wakelock
|
||||
wakeLock.release();
|
||||
addEditCourseDialog.setTitle(R.string.courses_edit_course);
|
||||
|
||||
TextView textView = (TextView) addEditCourseView.findViewById(R.id.edit_year);
|
||||
textView.setText(String.valueOf(course.getYear()));
|
||||
textView = (TextView) addEditCourseView.findViewById(R.id.edit_semester);
|
||||
textView.setText(String.valueOf(course.getCourseSemester()));
|
||||
textView = (TextView) addEditCourseView.findViewById(R.id.edit_section);
|
||||
textView.setText(String.valueOf(course.getCourseSection()));
|
||||
|
||||
addEditCourseDialog.show();
|
||||
editCourseListeners();
|
||||
}
|
||||
}
|
||||
|
||||
private class SaveEditCourse extends AsyncTask<Course, Course, Boolean> {
|
||||
private Context context;
|
||||
private PowerManager.WakeLock wakeLock;
|
||||
|
||||
protected SaveEditCourse(Context context) {
|
||||
this.context = context;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
super.onPreExecute();
|
||||
// Let's make sure the CPU doesn't go to sleep
|
||||
PowerManager powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, getClass().getName());
|
||||
if (!wakeLock.isHeld()) {
|
||||
wakeLock.acquire();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Boolean doInBackground(Course... courses) {
|
||||
for (Course course : courses) {
|
||||
if (isCancelled()) {
|
||||
return false;
|
||||
}
|
||||
|
||||
Cursor cursor = ubbDB.query(DBCourses.TABLE_NAME,
|
||||
DBCourses.ALL_COLUMNS, DBCourses.COLUMN_SECTION + "=" + course.getCourseSection() + " AND " + DBCourses.COLUMN_YEAR + "=" + course.getYear() + " AND " + DBCourses.COLUMN_SEMESTER + "=" + course.getCourseSemester() + " AND " + DBCourses.COLUMN_SUBJECT + "=" + course.getSubjectId(),
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
"1");
|
||||
if (cursor.getCount() == 0) {
|
||||
ContentValues values = new ContentValues();
|
||||
values.put(DBCourses.COLUMN_SECTION, course.getCourseSection());
|
||||
values.put(DBCourses.COLUMN_SEMESTER, course.getCourseSemester());
|
||||
values.put(DBCourses.COLUMN_YEAR, course.getYear());
|
||||
ubbDB.update(DBCourses.TABLE_NAME,
|
||||
values,
|
||||
DBCourses.COLUMN_ID + "=" + course.getCourseId(),
|
||||
null);
|
||||
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
}
|
||||
else {
|
||||
return true;
|
||||
}
|
||||
cursor.close();
|
||||
}
|
||||
publishProgress(courses);
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onProgressUpdate(Course... courses) {
|
||||
// Add the course to the adapter
|
||||
for (Course course : courses) {
|
||||
int index;
|
||||
for (index = 0; index < optionsSelected.length; index++) {
|
||||
if (optionsSelected[index]) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
courseListAdapter.updateCourse(index, course);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Boolean result) {
|
||||
// Release the kraken errr wakelock
|
||||
if (result) {
|
||||
Toast.makeText(context, R.string.courses_course_exists, Toast.LENGTH_SHORT).show();
|
||||
}
|
||||
else {
|
||||
Toast.makeText(context, R.string.courses_edited, Toast.LENGTH_SHORT).show();
|
||||
}
|
||||
wakeLock.release();
|
||||
addEditCourseDialog.dismiss();
|
||||
// Update the adapter list to show
|
||||
for (int i = 0; i < optionsSelected.length; i++) {
|
||||
if (optionsSelected[i]) {
|
||||
CardView cardView = (CardView) recyclerView.getChildAt(i);
|
||||
cardView.setCardBackgroundColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimaryDark));
|
||||
}
|
||||
}
|
||||
recyclerView.swapAdapter(courseListAdapter, true);
|
||||
optionsSelected = null;
|
||||
invalidateOptionsMenu();
|
||||
fab.show();
|
||||
eraseAddEditCourses();
|
||||
}
|
||||
}
|
||||
|
||||
private class AddCourse extends AsyncTask<Course, Course, Boolean> {
|
||||
private Context context;
|
||||
private PowerManager.WakeLock wakeLock;
|
||||
|
||||
protected AddCourse(Context context) {
|
||||
this.context = context;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
super.onPreExecute();
|
||||
// Let's make sure the CPU doesn't go to sleep
|
||||
PowerManager powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, getClass().getName());
|
||||
if (!wakeLock.isHeld()) {
|
||||
wakeLock.acquire();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Boolean doInBackground(Course... courses) {
|
||||
for (Course course : courses) {
|
||||
if (isCancelled()) {
|
||||
return false;
|
||||
}
|
||||
|
||||
Cursor cursor = ubbDB.query(DBCourses.TABLE_NAME,
|
||||
DBCourses.ALL_COLUMNS, DBCourses.COLUMN_SECTION + "=" + course.getCourseSection() + " AND " + DBCourses.COLUMN_YEAR + "=" + course.getYear() + " AND " + DBCourses.COLUMN_SEMESTER + "=" + course.getCourseSemester() + " AND " + DBCourses.COLUMN_SUBJECT + "=" + course.getSubjectId(),
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
"1");
|
||||
if (cursor.getCount() == 0) {
|
||||
ContentValues values = new ContentValues();
|
||||
values.put(DBCourses.COLUMN_SUBJECT, course.getSubjectId());
|
||||
values.put(DBCourses.COLUMN_SECTION, course.getCourseSection());
|
||||
values.put(DBCourses.COLUMN_SEMESTER, course.getCourseSemester());
|
||||
values.put(DBCourses.COLUMN_YEAR, course.getYear());
|
||||
ubbDB.insert(DBCourses.TABLE_NAME, null, values);
|
||||
}
|
||||
else {
|
||||
return true;
|
||||
}
|
||||
cursor.close();
|
||||
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
}
|
||||
publishProgress(courses);
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onProgressUpdate(Course... courses) {
|
||||
// Add the course to the adapter
|
||||
for (Course course : courses) {
|
||||
courseListAdapter.addCourse(course);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Boolean result) {
|
||||
// Release the kraken errr wakelock
|
||||
if (result) {
|
||||
Toast.makeText(context, R.string.courses_course_exists, Toast.LENGTH_SHORT).show();
|
||||
}
|
||||
else {
|
||||
Toast.makeText(context, R.string.courses_added, Toast.LENGTH_SHORT).show();
|
||||
}
|
||||
wakeLock.release();
|
||||
addEditCourseDialog.dismiss();
|
||||
// Update the adapter list to show
|
||||
recyclerView.swapAdapter(courseListAdapter, true);
|
||||
eraseAddEditCourses();
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,80 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.support.v7.widget.CardView;
|
||||
import android.support.v7.widget.RecyclerView;
|
||||
import android.view.LayoutInflater;
|
||||
import android.view.View;
|
||||
import android.view.ViewGroup;
|
||||
import android.widget.TextView;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
public class CourseListAdapter extends RecyclerView.Adapter<CourseListAdapter.CourseViewHolder> {
|
||||
|
||||
private List<Course> courses;
|
||||
|
||||
public CourseListAdapter(List<Course> courses) {
|
||||
this.courses = courses;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getItemCount() {
|
||||
return courses.size();
|
||||
}
|
||||
|
||||
@Override
|
||||
public CourseViewHolder onCreateViewHolder(ViewGroup viewGroup, int i) {
|
||||
View view = LayoutInflater.from(viewGroup.getContext()).inflate(R.layout.view_course_list_card, viewGroup, false);
|
||||
return new CourseViewHolder(view);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onBindViewHolder(CourseViewHolder courseViewHolder, int i) {
|
||||
courseViewHolder.courseYear.setText(String.valueOf(courses.get(i).getYear()));
|
||||
courseViewHolder.courseSemester.setText(String.valueOf(courses.get(i).getCourseSemester()));
|
||||
courseViewHolder.courseSection.setText(String.valueOf(courses.get(i).getCourseSection()));
|
||||
courseViewHolder.courseId.setText(String.valueOf(courses.get(i).getCourseId()));
|
||||
courseViewHolder.itemView.setLongClickable(true);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onAttachedToRecyclerView(RecyclerView recyclerView) {
|
||||
super.onAttachedToRecyclerView(recyclerView);
|
||||
}
|
||||
|
||||
public void addCourse(Course course) {
|
||||
courses.add(getItemCount(), course);
|
||||
notifyItemInserted(getItemCount());
|
||||
}
|
||||
|
||||
public void updateCourse(int index, Course course) {
|
||||
courses.set(index, course);
|
||||
notifyItemChanged(index);
|
||||
}
|
||||
|
||||
public Course getCourse(int index) {
|
||||
return courses.get(index);
|
||||
}
|
||||
|
||||
public void deleteCourse(Course course) {
|
||||
courses.remove(course);
|
||||
}
|
||||
|
||||
public static class CourseViewHolder extends RecyclerView.ViewHolder {
|
||||
public CardView cardView;
|
||||
public TextView courseYear;
|
||||
public TextView courseSemester;
|
||||
public TextView courseSection;
|
||||
public TextView courseId;
|
||||
|
||||
public CourseViewHolder(View itemView) {
|
||||
super(itemView);
|
||||
cardView = (CardView) itemView.findViewById(R.id.course_card_view);
|
||||
courseYear = (TextView) itemView.findViewById(R.id.course_year);
|
||||
courseSemester = (TextView) itemView.findViewById(R.id.course_semester);
|
||||
courseSection = (TextView) itemView.findViewById(R.id.course_section);
|
||||
courseId = (TextView) itemView.findViewById(R.id.course_id);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,867 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.content.ActivityNotFoundException;
|
||||
import android.content.Context;
|
||||
import android.content.DialogInterface;
|
||||
import android.content.Intent;
|
||||
import android.content.res.ColorStateList;
|
||||
import android.database.Cursor;
|
||||
import android.database.sqlite.SQLiteDatabase;
|
||||
import android.net.Uri;
|
||||
import android.os.AsyncTask;
|
||||
import android.os.Bundle;
|
||||
import android.os.Environment;
|
||||
import android.os.Handler;
|
||||
import android.os.PowerManager;
|
||||
import android.support.v4.content.ContextCompat;
|
||||
import android.support.v7.app.AlertDialog;
|
||||
import android.support.v7.app.AppCompatActivity;
|
||||
import android.util.Log;
|
||||
import android.view.LayoutInflater;
|
||||
import android.view.View;
|
||||
import android.widget.AdapterView;
|
||||
import android.widget.ArrayAdapter;
|
||||
import android.widget.RelativeLayout;
|
||||
import android.widget.Spinner;
|
||||
import android.widget.Toast;
|
||||
|
||||
import org.apache.poi.hssf.usermodel.HSSFWorkbook;
|
||||
import org.apache.poi.ss.usermodel.Cell;
|
||||
import org.apache.poi.ss.usermodel.Row;
|
||||
import org.apache.poi.ss.usermodel.Sheet;
|
||||
import org.apache.poi.ss.usermodel.Workbook;
|
||||
|
||||
import java.io.File;
|
||||
import java.io.FileNotFoundException;
|
||||
import java.io.FileOutputStream;
|
||||
import java.io.IOException;
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
import cl.cromer.ubb.attendance.Progress;
|
||||
import cl.cromer.ubb.attendance.DBSchema.*;
|
||||
|
||||
public class CourseReport extends AppCompatActivity {
|
||||
|
||||
// SQLite database
|
||||
private SQLParser sqlParser = null;
|
||||
private SQLiteDatabase ubbDB = null;
|
||||
|
||||
// Background thread for the database
|
||||
private Thread thread = null;
|
||||
private Handler threadHandler = new Handler();
|
||||
|
||||
// Progress bar
|
||||
private Progress progress = null;
|
||||
|
||||
private AlertDialog choiceDialog = null;
|
||||
private AlertDialog confirmDialog = null;
|
||||
private View reportView;
|
||||
|
||||
private List<Subject> subjects = new ArrayList<>();
|
||||
private List<Course> courses = new ArrayList<>();
|
||||
private List<Class> classes = new ArrayList<>();
|
||||
private List<Attendance> students = new ArrayList<>();
|
||||
|
||||
private Course course = new Course();
|
||||
|
||||
private int disabledButtonColor;
|
||||
|
||||
private Context context;
|
||||
|
||||
@Override
|
||||
protected void onCreate(Bundle savedInstanceState) {
|
||||
super.onCreate(savedInstanceState);
|
||||
setContentView(R.layout.activity_class_report);
|
||||
|
||||
this.context = this;
|
||||
|
||||
LayoutInflater inflater = (LayoutInflater) this.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
|
||||
reportView = inflater.inflate(R.layout.view_course_report, new RelativeLayout(this), false);
|
||||
|
||||
// Build the add subject dialog window using the subject view
|
||||
AlertDialog.Builder builder = new AlertDialog.Builder(this);
|
||||
builder.setView(reportView);
|
||||
builder.setPositiveButton(R.string.input_accept, null);
|
||||
builder.setNegativeButton(R.string.input_cancel, new DialogInterface.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(DialogInterface dialog, int which) {
|
||||
finish();
|
||||
}
|
||||
});
|
||||
|
||||
choiceDialog = builder.create();
|
||||
|
||||
choiceDialog.setOnShowListener(new DialogInterface.OnShowListener() {
|
||||
@Override
|
||||
public void onShow(DialogInterface dialogInterface) {
|
||||
ColorStateList colorStateList = choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).getTextColors();
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setEnabled(false);
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setClickable(false);
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_NEGATIVE).setTextColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimaryDark));
|
||||
|
||||
if (colorStateList != null) {
|
||||
disabledButtonColor = colorStateList.getColorForState(new int[] {-android.R.attr.state_enabled}, R.color.colorPrimary);
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
choiceDialog.getWindow().getAttributes().windowAnimations = R.style.DialogAnimation;
|
||||
choiceDialog.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
finish();
|
||||
}
|
||||
});
|
||||
choiceDialog.setCanceledOnTouchOutside(false);
|
||||
|
||||
// Create a progress dialog for slow devices
|
||||
progress = new Progress();
|
||||
progress.show(this, 1);
|
||||
progress.setCancelable(false);
|
||||
|
||||
// Load the SQLite database
|
||||
sqlParser = new SQLParser(this);
|
||||
thread = new Thread(new Runnable() {
|
||||
public void run() {
|
||||
ubbDB = sqlParser.getWritableDatabase();
|
||||
threadHandler.post(new Runnable() {
|
||||
public void run() {
|
||||
databaseLoaded();
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
thread.start();
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
thread.interrupt();
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onDestroy() {
|
||||
super.onDestroy();
|
||||
// We need to get rid of the progressbar if it's showing
|
||||
if (progress != null && progress.isShowing()) {
|
||||
progress.dismiss();
|
||||
}
|
||||
if (choiceDialog != null && choiceDialog.isShowing()) {
|
||||
choiceDialog.dismiss();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onBackPressed() {
|
||||
super.onBackPressed();
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onSupportNavigateUp() {
|
||||
onBackPressed();
|
||||
return true;
|
||||
}
|
||||
|
||||
private void databaseLoaded() {
|
||||
final GetSubjects getSubjects = new GetSubjects(getApplicationContext());
|
||||
getSubjects.execute();
|
||||
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
getSubjects.cancel(true);
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private AdapterView.OnItemSelectedListener subjectListener() {
|
||||
return new AdapterView.OnItemSelectedListener() {
|
||||
@Override
|
||||
public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {
|
||||
courses = new ArrayList<>();
|
||||
if (position == 0) {
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setTextColor(disabledButtonColor);
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setEnabled(false);
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setClickable(false);
|
||||
|
||||
Spinner spinner = (Spinner) reportView.findViewById(R.id.course_spinner);
|
||||
|
||||
List<String> options = new ArrayList<>();
|
||||
options.add(getResources().getString(R.string.report_course));
|
||||
|
||||
ArrayAdapter<String> arrayAdapter = new ArrayAdapter<>(getApplicationContext(), R.layout.view_spinner_selected_item, options);
|
||||
arrayAdapter.setDropDownViewResource(R.layout.view_spinner_dropdown_item);
|
||||
|
||||
spinner.setAdapter(arrayAdapter);
|
||||
spinner.setSelection(0);
|
||||
spinner.setEnabled(false);
|
||||
spinner.setClickable(false);
|
||||
spinner.setOnItemSelectedListener(null);
|
||||
}
|
||||
else {
|
||||
progress = new Progress();
|
||||
progress.show(context, 1);
|
||||
progress.setCancelable(false);
|
||||
|
||||
final int thePosition = position - 1;
|
||||
|
||||
course.setSubject(subjects.get(thePosition));
|
||||
|
||||
sqlParser = new SQLParser(getApplicationContext());
|
||||
thread = new Thread(new Runnable() {
|
||||
public void run() {
|
||||
ubbDB = sqlParser.getWritableDatabase();
|
||||
threadHandler.post(new Runnable() {
|
||||
public void run() {
|
||||
|
||||
final GetCourses getCourses = new GetCourses(getApplicationContext());
|
||||
getCourses.execute(thePosition);
|
||||
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
getCourses.cancel(true);
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
thread.start();
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
thread.interrupt();
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onNothingSelected(AdapterView<?> parent) {
|
||||
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
private AdapterView.OnItemSelectedListener courseListener() {
|
||||
return new AdapterView.OnItemSelectedListener() {
|
||||
@Override
|
||||
public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {
|
||||
if (position == 0) {
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setTextColor(disabledButtonColor);
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setEnabled(false);
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setClickable(false);
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setOnClickListener(null);
|
||||
}
|
||||
else {
|
||||
course.setCourse(courses.get(position - 1));
|
||||
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setTextColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimaryDark));
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setEnabled(true);
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setClickable(true);
|
||||
choiceDialog.getButton(AlertDialog.BUTTON_POSITIVE).setOnClickListener(acceptListener());
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onNothingSelected(AdapterView<?> parent) {
|
||||
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
private View.OnClickListener acceptListener() {
|
||||
return new View.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(View view) {
|
||||
progress = new Progress();
|
||||
progress.show(context, 1);
|
||||
progress.setCancelable(false);
|
||||
|
||||
choiceDialog.dismiss();
|
||||
|
||||
sqlParser = new SQLParser(getApplicationContext());
|
||||
thread = new Thread(new Runnable() {
|
||||
public void run() {
|
||||
ubbDB = sqlParser.getWritableDatabase();
|
||||
threadHandler.post(new Runnable() {
|
||||
public void run() {
|
||||
|
||||
final GetClasses getClasses = new GetClasses(getApplicationContext());
|
||||
getClasses.execute();
|
||||
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
getClasses.cancel(true);
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
thread.start();
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
thread.interrupt();
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
private class GetSubjects extends AsyncTask<Void, Void, Void> {
|
||||
private Context context;
|
||||
private PowerManager.WakeLock wakeLock;
|
||||
|
||||
protected GetSubjects(Context context) {
|
||||
this.context = context;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
super.onPreExecute();
|
||||
// Let's make sure the CPU doesn't go to sleep
|
||||
PowerManager powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, getClass().getName());
|
||||
if (!wakeLock.isHeld()) {
|
||||
wakeLock.acquire();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Void doInBackground(Void... voids) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
// Get subjects by name in ascending order
|
||||
String query = "SELECT " + sqlParser.combineColumnsStrings(DBSubjects.ALL_COLUMNS, DBMajors.ALL_COLUMNS) + " FROM " + DBSubjects.TABLE_NAME + " INNER JOIN (SELECT " + sqlParser.combineColumnsStrings(DBMajors.ALL_COLUMNS) +" FROM " + DBMajors.TABLE_NAME + ") " + DBMajors.TABLE_NAME + " ON " + DBSubjects.TABLE_NAME + "." + DBSubjects.COLUMN_MAJOR + "=" + DBMajors.TABLE_NAME + "." + DBMajors.COLUMN_ID + " ORDER BY " + DBSubjects.COLUMN_NAME + " ASC";
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.d("CourseReport", query);
|
||||
}
|
||||
Cursor cursor = ubbDB.rawQuery(query, null);
|
||||
|
||||
// Iterate through the database rows
|
||||
while (cursor.moveToNext()) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
Major major = new Major(
|
||||
cursor.getInt(cursor.getColumnIndex(DBMajors.COLUMN_ID)),
|
||||
cursor.getString(cursor.getColumnIndex(DBMajors.COLUMN_NAME)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBMajors.COLUMN_CODE)));
|
||||
subjects.add(
|
||||
new Subject(
|
||||
major,
|
||||
cursor.getInt(cursor.getColumnIndex(DBSubjects.COLUMN_ID)),
|
||||
cursor.getString(cursor.getColumnIndex(DBSubjects.COLUMN_NAME)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBSubjects.COLUMN_CODE))
|
||||
)
|
||||
);
|
||||
}
|
||||
cursor.close();
|
||||
|
||||
// Close the database connection
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Void result) {
|
||||
// Release the kraken errr wakelock
|
||||
List<String> options = new ArrayList<>();
|
||||
options.add(getResources().getString(R.string.report_subject));
|
||||
for (Subject subject : subjects) {
|
||||
options.add(subject.getSubjectName());
|
||||
}
|
||||
|
||||
Spinner spinner = (Spinner) reportView.findViewById(R.id.subject_spinner);
|
||||
|
||||
ArrayAdapter<String> arrayAdapter = new ArrayAdapter<>(getApplicationContext(), R.layout.view_spinner_selected_item, options);
|
||||
arrayAdapter.setDropDownViewResource(R.layout.view_spinner_dropdown_item);
|
||||
|
||||
spinner.setAdapter(arrayAdapter);
|
||||
spinner.setSelection(0);
|
||||
|
||||
spinner.setOnItemSelectedListener(subjectListener());
|
||||
progress.dismiss();
|
||||
wakeLock.release();
|
||||
choiceDialog.show();
|
||||
}
|
||||
}
|
||||
|
||||
private class GetCourses extends AsyncTask<Integer, Void, Void> {
|
||||
private Context context;
|
||||
private PowerManager.WakeLock wakeLock;
|
||||
|
||||
protected GetCourses(Context context) {
|
||||
this.context = context;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
super.onPreExecute();
|
||||
// Let's make sure the CPU doesn't go to sleep
|
||||
PowerManager powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, getClass().getName());
|
||||
if (!wakeLock.isHeld()) {
|
||||
wakeLock.acquire();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Void doInBackground(Integer... position) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
Cursor cursor = ubbDB.query(
|
||||
DBCourses.TABLE_NAME,
|
||||
DBCourses.ALL_COLUMNS,
|
||||
DBCourses.COLUMN_SUBJECT + "=" + subjects.get(position[0]).getSubjectId(),
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
DBCourses.COLUMN_YEAR + " DESC, " + DBCourses.COLUMN_SEMESTER + " DESC, " + DBCourses.COLUMN_SECTION + " ASC",
|
||||
null);
|
||||
|
||||
// Iterate through the database rows
|
||||
while (cursor.moveToNext()) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
courses.add(
|
||||
new Course(
|
||||
subjects.get(position[0]),
|
||||
cursor.getInt(cursor.getColumnIndex(DBCourses.COLUMN_ID)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBCourses.COLUMN_SECTION)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBCourses.COLUMN_SEMESTER)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBCourses.COLUMN_YEAR))
|
||||
)
|
||||
);
|
||||
}
|
||||
cursor.close();
|
||||
|
||||
// Close the database connection
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Void result) {
|
||||
// Release the kraken errr wakelock
|
||||
List<String> options = new ArrayList<>();
|
||||
options.add(getResources().getString(R.string.report_course));
|
||||
for (Course course : courses) {
|
||||
options.add(course.getYear() + "-" + course.getCourseSemester() + " " + getResources().getString(R.string.report_section) + " " + course.getCourseSection());
|
||||
}
|
||||
|
||||
Spinner spinner = (Spinner) reportView.findViewById(R.id.course_spinner);
|
||||
|
||||
ArrayAdapter<String> arrayAdapter = new ArrayAdapter<>(getApplicationContext(), R.layout.view_spinner_selected_item, options);
|
||||
arrayAdapter.setDropDownViewResource(R.layout.view_spinner_dropdown_item);
|
||||
|
||||
spinner.setAdapter(arrayAdapter);
|
||||
spinner.setSelection(0);
|
||||
spinner.setEnabled(true);
|
||||
spinner.setClickable(true);
|
||||
|
||||
spinner.setOnItemSelectedListener(courseListener());
|
||||
progress.dismiss();
|
||||
wakeLock.release();
|
||||
}
|
||||
}
|
||||
|
||||
private class GetClasses extends AsyncTask<Integer, Void, Boolean> {
|
||||
private Context context;
|
||||
private PowerManager.WakeLock wakeLock;
|
||||
|
||||
protected GetClasses(Context context) {
|
||||
this.context = context;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
super.onPreExecute();
|
||||
// Let's make sure the CPU doesn't go to sleep
|
||||
PowerManager powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, getClass().getName());
|
||||
if (!wakeLock.isHeld()) {
|
||||
wakeLock.acquire();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Boolean doInBackground(Integer... position) {
|
||||
if (isCancelled()) {
|
||||
return false;
|
||||
}
|
||||
|
||||
Cursor cursor = ubbDB.query(
|
||||
DBClasses.TABLE_NAME,
|
||||
DBClasses.ALL_COLUMNS,
|
||||
DBClasses.COLUMN_COURSE + "=" + course.getCourseId(),
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
DBClasses.COLUMN_DATE + " DESC",
|
||||
null);
|
||||
|
||||
// Iterate through the database rows
|
||||
while (cursor.moveToNext()) {
|
||||
if (isCancelled()) {
|
||||
return false;
|
||||
}
|
||||
classes.add(
|
||||
new Class(
|
||||
course,
|
||||
cursor.getInt(cursor.getColumnIndex(DBClasses.COLUMN_ID)),
|
||||
cursor.getLong(cursor.getColumnIndex(DBClasses.COLUMN_DATE))
|
||||
)
|
||||
);
|
||||
}
|
||||
cursor.close();
|
||||
|
||||
String query = "SELECT " + sqlParser.combineColumnsStrings(DBStudents.ALL_COLUMNS) + " FROM " + DBStudents.TABLE_NAME +
|
||||
" INNER JOIN (SELECT " + sqlParser.combineColumnsStrings(DBCoursesStudents.ALL_COLUMNS) + " FROM " + DBCoursesStudents.TABLE_NAME + " WHERE " + DBCoursesStudents.COLUMN_COURSE + "=" + course.getCourseId() + ") AS " + DBCoursesStudents.TABLE_NAME + " ON " + DBStudents.TABLE_NAME + "." + DBStudents.COLUMN_ID + "=" + DBCoursesStudents.TABLE_NAME + "." + DBCoursesStudents.COLUMN_STUDENT +
|
||||
" ORDER BY " + DBStudents.COLUMN_FIRST_LAST_NAME + " ASC, " + DBStudents.COLUMN_FIRST_NAME + " ASC";
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.d("CourseReport", query);
|
||||
}
|
||||
cursor = ubbDB.rawQuery(query, null);
|
||||
|
||||
// Iterate through the database rows
|
||||
while (cursor.moveToNext()) {
|
||||
if (isCancelled()) {
|
||||
return false;
|
||||
}
|
||||
students.add(new Attendance(
|
||||
0,
|
||||
cursor.getInt(cursor.getColumnIndex(DBStudents.COLUMN_ID)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_RUN)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_FIRST_NAME)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_SECOND_NAME)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_FIRST_LAST_NAME)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_SECOND_LAST_NAME)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBStudents.COLUMN_MAJOR)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBStudents.COLUMN_ENROLLED)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_EMAIL)),
|
||||
cursor.getBlob(cursor.getColumnIndex(DBStudents.COLUMN_PHOTO))
|
||||
));
|
||||
}
|
||||
cursor.close();
|
||||
|
||||
for (Class classObject : classes) {
|
||||
if (isCancelled()) {
|
||||
return false;
|
||||
}
|
||||
|
||||
cursor = ubbDB.query(
|
||||
DBAttendance.TABLE_NAME,
|
||||
DBAttendance.ALL_COLUMNS,
|
||||
DBAttendance.COLUMN_CLASS + "=" + classObject.getClassId(),
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
null);
|
||||
|
||||
while (cursor.moveToNext()) {
|
||||
if (isCancelled()) {
|
||||
return false;
|
||||
}
|
||||
|
||||
int studentId = cursor.getInt(cursor.getColumnIndex(DBAttendance.COLUMN_STUDENT));
|
||||
int status = cursor.getInt(cursor.getColumnIndex(DBAttendance.COLUMN_ATTENDANCE));
|
||||
|
||||
for (int i = 0; i < students.size(); i++) {
|
||||
if (students.get(i).getStudentId() == studentId) {
|
||||
// 0 - uninitialized
|
||||
// 1 - present
|
||||
// 2 - justified
|
||||
// 3 - absent
|
||||
// 4 - late
|
||||
switch (status) {
|
||||
case 1:
|
||||
students.get(i).setPresent(students.get(i).getPresent() + 1);
|
||||
break;
|
||||
case 2:
|
||||
students.get(i).setJustified(students.get(i).getJustified() + 1);
|
||||
break;
|
||||
case 3:
|
||||
students.get(i).setAbsent(students.get(i).getAbsent() + 1);
|
||||
break;
|
||||
case 4:
|
||||
students.get(i).setLate(students.get(i).getLate() + 1);
|
||||
break;
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
cursor.close();
|
||||
}
|
||||
|
||||
Workbook workbook = new HSSFWorkbook();
|
||||
Sheet sheet = workbook.createSheet(getResources().getString(R.string.app_name));
|
||||
|
||||
Excel excel = new Excel(workbook);
|
||||
excel.createStyles();
|
||||
|
||||
Row row;
|
||||
Cell cell;
|
||||
// Major
|
||||
row = sheet.createRow(0);
|
||||
cell = row.createCell(0);
|
||||
cell.setCellValue(course.getMajorName() + " - " + course.getMajorCode());
|
||||
cell.setCellStyle(excel.getCellStyle("left"));
|
||||
for (int i = 1; i < 7; i++) {
|
||||
cell = row.createCell(i);
|
||||
if (i == 6) {
|
||||
cell.setCellStyle(excel.getCellStyle("right"));
|
||||
}
|
||||
else {
|
||||
cell.setCellStyle(excel.getCellStyle("middle"));
|
||||
}
|
||||
}
|
||||
|
||||
// Subject
|
||||
row = sheet.createRow(1);
|
||||
cell = row.createCell(0);
|
||||
cell.setCellValue(course.getSubjectName() + " - " + course.getSubjectCode());
|
||||
cell.setCellStyle(excel.getCellStyle("left"));
|
||||
for (int i = 1; i < 7; i++) {
|
||||
cell = row.createCell(i);
|
||||
if (i == 6) {
|
||||
cell.setCellStyle(excel.getCellStyle("right"));
|
||||
}
|
||||
else {
|
||||
cell.setCellStyle(excel.getCellStyle("middle"));
|
||||
}
|
||||
}
|
||||
|
||||
// Number of classes
|
||||
row = sheet.createRow(2);
|
||||
cell = row.createCell(0);
|
||||
cell.setCellValue(classes.size() + " " + getResources().getString(R.string.report_classes));
|
||||
cell.setCellStyle(excel.getCellStyle("left"));
|
||||
for (int i = 1; i < 7; i++) {
|
||||
cell = row.createCell(i);
|
||||
if (i == 6) {
|
||||
cell.setCellStyle(excel.getCellStyle("right"));
|
||||
}
|
||||
else {
|
||||
cell.setCellStyle(excel.getCellStyle("middle"));
|
||||
}
|
||||
}
|
||||
|
||||
// Columns
|
||||
row = sheet.createRow(3);
|
||||
|
||||
cell = row.createCell(0);
|
||||
cell.setCellValue(getResources().getString(R.string.report_names));
|
||||
cell.setCellStyle(excel.getCellStyle("blue"));
|
||||
|
||||
cell = row.createCell(1);
|
||||
cell.setCellValue(getResources().getString(R.string.report_first_last_name));
|
||||
cell.setCellStyle(excel.getCellStyle("blue"));
|
||||
|
||||
cell = row.createCell(2);
|
||||
cell.setCellValue(getResources().getString(R.string.report_second_last_name));
|
||||
cell.setCellStyle(excel.getCellStyle("blue"));
|
||||
|
||||
cell = row.createCell(3);
|
||||
cell.setCellValue(getResources().getString(R.string.report_present));
|
||||
cell.setCellStyle(excel.getCellStyle("green"));
|
||||
|
||||
cell = row.createCell(4);
|
||||
cell.setCellValue(getResources().getString(R.string.report_late));
|
||||
cell.setCellStyle(excel.getCellStyle("yellow"));
|
||||
|
||||
cell = row.createCell(5);
|
||||
cell.setCellValue(getResources().getString(R.string.report_justified));
|
||||
cell.setCellStyle(excel.getCellStyle("orange"));
|
||||
|
||||
cell = row.createCell(6);
|
||||
cell.setCellValue(getResources().getString(R.string.report_absent));
|
||||
cell.setCellStyle(excel.getCellStyle("red"));
|
||||
|
||||
int cellSize1 = getResources().getString(R.string.report_names).length();
|
||||
int cellSize2 = getResources().getString(R.string.report_first_last_name).length();
|
||||
int cellSize3 = getResources().getString(R.string.report_second_last_name).length();
|
||||
int rowNumber = 4;
|
||||
String value;
|
||||
for (Attendance student : students) {
|
||||
row = sheet.createRow(rowNumber);
|
||||
|
||||
// First and second name
|
||||
cell = row.createCell(0);
|
||||
value = student.getFirstName() + " " + student.getSecondName();
|
||||
if (value.length() > cellSize1) {
|
||||
cellSize1 = value.length();
|
||||
}
|
||||
cell.setCellValue(value);
|
||||
cell.setCellStyle(excel.getCellStyle("main"));
|
||||
|
||||
// First last name
|
||||
cell = row.createCell(1);
|
||||
value = student.getFirstLastName();
|
||||
if (value.length() > cellSize2) {
|
||||
cellSize2 = value.length();
|
||||
}
|
||||
cell.setCellValue(value);
|
||||
cell.setCellStyle(excel.getCellStyle("main"));
|
||||
|
||||
// Second last name
|
||||
cell = row.createCell(2);
|
||||
value = student.getSecondLastName();
|
||||
if (value.length() > cellSize3) {
|
||||
cellSize3 = value.length();
|
||||
}
|
||||
cell.setCellValue(value);
|
||||
cell.setCellStyle(excel.getCellStyle("main"));
|
||||
|
||||
for (int i = 3; i < 7; i++) {
|
||||
cell = row.createCell(i);
|
||||
switch (i) {
|
||||
case 3:
|
||||
cell.setCellStyle(excel.getCellStyle("green"));
|
||||
cell.setCellValue(student.getPresent());
|
||||
break;
|
||||
case 4:
|
||||
cell.setCellStyle(excel.getCellStyle("yellow"));
|
||||
cell.setCellValue(student.getLate());
|
||||
break;
|
||||
case 5:
|
||||
cell.setCellStyle(excel.getCellStyle("orange"));
|
||||
cell.setCellValue(student.getJustified());
|
||||
break;
|
||||
case 6:
|
||||
cell.setCellStyle(excel.getCellStyle("red"));
|
||||
cell.setCellValue(student.getAbsent());
|
||||
break;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
rowNumber++;
|
||||
}
|
||||
sheet.setColumnWidth(0, (cellSize1 + 1) * 256);
|
||||
sheet.setColumnWidth(1, (cellSize2 + 1) * 256);
|
||||
sheet.setColumnWidth(2, (cellSize3 + 1) * 256);
|
||||
|
||||
final String filePath = Environment.getExternalStorageDirectory() +
|
||||
"/" + getResources().getString(R.string.app_name) +
|
||||
"/" + course.getMajorName();
|
||||
|
||||
final String fileName = "/" + getResources().getString(R.string.report_course) + "-" + course.getYear() + "-" + course.getCourseSemester() + "-" + course.getCourseSection() + ".xls";
|
||||
|
||||
try {
|
||||
File file = new File(filePath);
|
||||
if (!file.exists()) {
|
||||
if (!file.mkdirs()) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
FileOutputStream fileOutputStream = new FileOutputStream(filePath + fileName);
|
||||
workbook.write(fileOutputStream);
|
||||
fileOutputStream.close();
|
||||
}
|
||||
catch (FileNotFoundException e) {
|
||||
if (BuildConfig.DEBUG) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
return false;
|
||||
}
|
||||
catch (IOException e) {
|
||||
if (BuildConfig.DEBUG) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
// Close the database connection
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Boolean result) {
|
||||
// Release the kraken errr wakelock
|
||||
if (result) {
|
||||
confirmOpen();
|
||||
}
|
||||
else {
|
||||
Toast.makeText(getApplicationContext(), R.string.report_fail, Toast.LENGTH_SHORT).show();
|
||||
finish();
|
||||
}
|
||||
progress.dismiss();
|
||||
wakeLock.release();
|
||||
}
|
||||
}
|
||||
|
||||
private void confirmOpen() {
|
||||
final String filePath = Environment.getExternalStorageDirectory() +
|
||||
"/" + getResources().getString(R.string.app_name) +
|
||||
"/" + course.getMajorName();
|
||||
|
||||
final String fileName = "/" + getResources().getString(R.string.report_course) + "-" + course.getYear() + "-" + course.getCourseSemester() + "-" + course.getCourseSection() + ".xls";
|
||||
|
||||
AlertDialog.Builder builder = new AlertDialog.Builder(context);
|
||||
builder.setMessage(R.string.report_created);
|
||||
builder.setPositiveButton(R.string.input_yes, new DialogInterface.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(DialogInterface dialog, int which) {
|
||||
File file = new File(filePath + fileName);
|
||||
Intent intent = new Intent(Intent.ACTION_VIEW);
|
||||
intent.setDataAndType(Uri.fromFile(file), "application/vnd.ms-excel");
|
||||
|
||||
try {
|
||||
startActivity(intent);
|
||||
}
|
||||
catch (ActivityNotFoundException e) {
|
||||
Toast.makeText(CourseReport.this, "No Application Available to View Excel", Toast.LENGTH_SHORT).show();
|
||||
}
|
||||
confirmDialog.dismiss();
|
||||
finish();
|
||||
}
|
||||
});
|
||||
builder.setNegativeButton(R.string.input_no, new DialogInterface.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(DialogInterface dialog, int which) {
|
||||
confirmDialog.dismiss();
|
||||
finish();
|
||||
}
|
||||
});
|
||||
|
||||
builder.setCancelable(true);
|
||||
confirmDialog = builder.create();
|
||||
|
||||
confirmDialog.setOnShowListener(new DialogInterface.OnShowListener() {
|
||||
@Override
|
||||
public void onShow(DialogInterface dialogInterface) {
|
||||
confirmDialog.getButton(AlertDialog.BUTTON_POSITIVE).setTextColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimaryDark));
|
||||
confirmDialog.getButton(AlertDialog.BUTTON_NEGATIVE).setTextColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimaryDark));
|
||||
}
|
||||
});
|
||||
|
||||
confirmDialog.show();
|
||||
}
|
||||
}
|
@ -0,0 +1,80 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.provider.BaseColumns;
|
||||
|
||||
final public class DBSchema {
|
||||
|
||||
// Disable the constructor
|
||||
private DBSchema() {}
|
||||
|
||||
protected static abstract class DBMajors implements BaseColumns {
|
||||
protected static final String TABLE_NAME = "majors";
|
||||
protected static final String COLUMN_ID = "major_id";
|
||||
protected static final String COLUMN_CODE = "major_code";
|
||||
protected static final String COLUMN_NAME = "major_name";
|
||||
protected static final String[] ALL_COLUMNS = {COLUMN_ID, COLUMN_CODE, COLUMN_NAME};
|
||||
}
|
||||
|
||||
protected static abstract class DBSubjects implements BaseColumns {
|
||||
protected static final String TABLE_NAME = "subjects";
|
||||
protected static final String COLUMN_ID = "subject_id";
|
||||
protected static final String COLUMN_CODE = "subject_code";
|
||||
protected static final String COLUMN_NAME = "subject_name";
|
||||
protected static final String COLUMN_MAJOR = "subject_major";
|
||||
protected static final String[] ALL_COLUMNS = {COLUMN_ID, COLUMN_CODE, COLUMN_NAME, COLUMN_MAJOR};
|
||||
}
|
||||
|
||||
protected static abstract class DBCourses implements BaseColumns {
|
||||
protected static final String TABLE_NAME = "courses";
|
||||
protected static final String COLUMN_ID = "course_id";
|
||||
protected static final String COLUMN_SUBJECT = "course_subject";
|
||||
protected static final String COLUMN_SECTION = "course_section";
|
||||
protected static final String COLUMN_SEMESTER = "course_semester";
|
||||
protected static final String COLUMN_YEAR = "course_year";
|
||||
protected static final String[] ALL_COLUMNS = {COLUMN_ID, COLUMN_SUBJECT, COLUMN_SECTION, COLUMN_SEMESTER, COLUMN_YEAR};
|
||||
}
|
||||
|
||||
protected static abstract class DBCoursesStudents implements BaseColumns {
|
||||
protected static final String TABLE_NAME = "courses_students";
|
||||
protected static final String COLUMN_ID = "cs_id";
|
||||
protected static final String COLUMN_COURSE = "cs_course";
|
||||
protected static final String COLUMN_STUDENT = "cs_student";
|
||||
protected static final String[] ALL_COLUMNS = {COLUMN_ID, COLUMN_COURSE, COLUMN_STUDENT};
|
||||
}
|
||||
|
||||
protected static abstract class DBClasses implements BaseColumns {
|
||||
protected static final String TABLE_NAME = "classes";
|
||||
protected static final String COLUMN_ID = "class_id";
|
||||
protected static final String COLUMN_COURSE = "class_course";
|
||||
protected static final String COLUMN_DATE = "class_date";
|
||||
protected static final String[] ALL_COLUMNS = {COLUMN_ID, COLUMN_COURSE, COLUMN_DATE};
|
||||
}
|
||||
|
||||
protected static abstract class DBStudents implements BaseColumns {
|
||||
protected static final String TABLE_NAME = "students";
|
||||
protected static final String COLUMN_ID = "student_id";
|
||||
protected static final String COLUMN_RUN = "student_run";
|
||||
protected static final String COLUMN_FIRST_NAME = "student_first_name";
|
||||
protected static final String COLUMN_SECOND_NAME = "student_second_name";
|
||||
protected static final String COLUMN_FIRST_LAST_NAME = "student_first_last_name";
|
||||
protected static final String COLUMN_SECOND_LAST_NAME = "student_second_last_name";
|
||||
protected static final String COLUMN_FIRST_NAME_NORM = "student_first_name_norm";
|
||||
protected static final String COLUMN_SECOND_NAME_NORM = "student_second_name_norm";
|
||||
protected static final String COLUMN_FIRST_LAST_NAME_NORM = "student_first_last_name_norm";
|
||||
protected static final String COLUMN_SECOND_LAST_NAME_NORM = "student_second_last_name_norm";
|
||||
protected static final String COLUMN_MAJOR = "student_major";
|
||||
protected static final String COLUMN_EMAIL = "student_email";
|
||||
protected static final String COLUMN_ENROLLED = "student_enrolled";
|
||||
protected static final String COLUMN_PHOTO = "student_photo";
|
||||
protected static final String[] ALL_COLUMNS = {COLUMN_ID, COLUMN_RUN, COLUMN_FIRST_NAME, COLUMN_SECOND_NAME, COLUMN_FIRST_LAST_NAME, COLUMN_SECOND_LAST_NAME, COLUMN_MAJOR, COLUMN_EMAIL, COLUMN_ENROLLED, COLUMN_PHOTO};
|
||||
}
|
||||
|
||||
protected static abstract class DBAttendance implements BaseColumns {
|
||||
protected static final String TABLE_NAME = "attendance";
|
||||
protected static final String COLUMN_ID = "attendance_id";
|
||||
protected static final String COLUMN_CLASS = "attendance_class";
|
||||
protected static final String COLUMN_STUDENT = "attendance_student";
|
||||
protected static final String COLUMN_ATTENDANCE = "attendance_attendance";
|
||||
protected static final String[] ALL_COLUMNS = {COLUMN_ID, COLUMN_CLASS, COLUMN_STUDENT, COLUMN_ATTENDANCE};
|
||||
}
|
||||
}
|
171
attendance-ubb/src/main/java/cl/cromer/ubb/attendance/Excel.java
Normal file
@ -0,0 +1,171 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import org.apache.poi.hssf.util.HSSFColor;
|
||||
import org.apache.poi.ss.usermodel.CellStyle;
|
||||
import org.apache.poi.ss.usermodel.Font;
|
||||
import org.apache.poi.ss.usermodel.IndexedColors;
|
||||
import org.apache.poi.ss.usermodel.Workbook;
|
||||
|
||||
public class Excel {
|
||||
private Workbook workbook;
|
||||
|
||||
private CellStyle mainCellStyle = null;
|
||||
private CellStyle leftCellStyle = null;
|
||||
private CellStyle rightCellStyle = null;
|
||||
private CellStyle middleCellStyle = null;
|
||||
private CellStyle blueCellStyle = null;
|
||||
private CellStyle greenCellStyle = null;
|
||||
private CellStyle yellowCellStyle = null;
|
||||
private CellStyle orangeCellStyle = null;
|
||||
private CellStyle redCellStyle = null;
|
||||
|
||||
public Excel(Workbook workbook) {
|
||||
this.workbook = workbook;
|
||||
}
|
||||
|
||||
protected void createStyles() {
|
||||
Font font = workbook.createFont();
|
||||
mainCellStyle = workbook.createCellStyle();
|
||||
mainCellStyle.setBorderBottom(CellStyle.BORDER_THIN);
|
||||
mainCellStyle.setBottomBorderColor(IndexedColors.BLACK.getIndex());
|
||||
mainCellStyle.setBorderLeft(CellStyle.BORDER_THIN);
|
||||
mainCellStyle.setLeftBorderColor(IndexedColors.BLACK.getIndex());
|
||||
mainCellStyle.setBorderRight(CellStyle.BORDER_THIN);
|
||||
mainCellStyle.setRightBorderColor(IndexedColors.BLACK.getIndex());
|
||||
mainCellStyle.setBorderTop(CellStyle.BORDER_THIN);
|
||||
mainCellStyle.setTopBorderColor(IndexedColors.BLACK.getIndex());
|
||||
|
||||
leftCellStyle = workbook.createCellStyle();
|
||||
leftCellStyle.setBorderBottom(CellStyle.BORDER_THIN);
|
||||
leftCellStyle.setBottomBorderColor(IndexedColors.BLACK.getIndex());
|
||||
leftCellStyle.setBorderLeft(CellStyle.BORDER_THIN);
|
||||
leftCellStyle.setLeftBorderColor(IndexedColors.BLACK.getIndex());
|
||||
leftCellStyle.setBorderTop(CellStyle.BORDER_THIN);
|
||||
leftCellStyle.setTopBorderColor(IndexedColors.BLACK.getIndex());
|
||||
leftCellStyle.setFillForegroundColor(IndexedColors.BLUE.getIndex());
|
||||
leftCellStyle.setFillPattern(CellStyle.SOLID_FOREGROUND);
|
||||
font.setColor(HSSFColor.WHITE.index);
|
||||
leftCellStyle.setFont(font);
|
||||
|
||||
rightCellStyle = workbook.createCellStyle();
|
||||
rightCellStyle.setBorderBottom(CellStyle.BORDER_THIN);
|
||||
rightCellStyle.setBottomBorderColor(IndexedColors.BLACK.getIndex());
|
||||
rightCellStyle.setBorderRight(CellStyle.BORDER_THIN);
|
||||
rightCellStyle.setRightBorderColor(IndexedColors.BLACK.getIndex());
|
||||
rightCellStyle.setBorderTop(CellStyle.BORDER_THIN);
|
||||
rightCellStyle.setTopBorderColor(IndexedColors.BLACK.getIndex());
|
||||
rightCellStyle.setFillForegroundColor(IndexedColors.BLUE.getIndex());
|
||||
rightCellStyle.setFillPattern(CellStyle.SOLID_FOREGROUND);
|
||||
font.setColor(HSSFColor.WHITE.index);
|
||||
rightCellStyle.setFont(font);
|
||||
|
||||
middleCellStyle = workbook.createCellStyle();
|
||||
middleCellStyle.setBorderBottom(CellStyle.BORDER_THIN);
|
||||
middleCellStyle.setBottomBorderColor(IndexedColors.BLACK.getIndex());
|
||||
middleCellStyle.setBorderTop(CellStyle.BORDER_THIN);
|
||||
middleCellStyle.setTopBorderColor(IndexedColors.BLACK.getIndex());
|
||||
middleCellStyle.setFillForegroundColor(IndexedColors.BLUE.getIndex());
|
||||
middleCellStyle.setFillPattern(CellStyle.SOLID_FOREGROUND);
|
||||
font.setColor(HSSFColor.WHITE.index);
|
||||
middleCellStyle.setFont(font);
|
||||
|
||||
blueCellStyle = workbook.createCellStyle();
|
||||
blueCellStyle.setBorderBottom(CellStyle.BORDER_THIN);
|
||||
blueCellStyle.setBottomBorderColor(IndexedColors.BLACK.getIndex());
|
||||
blueCellStyle.setBorderLeft(CellStyle.BORDER_THIN);
|
||||
blueCellStyle.setLeftBorderColor(IndexedColors.BLACK.getIndex());
|
||||
blueCellStyle.setBorderRight(CellStyle.BORDER_THIN);
|
||||
blueCellStyle.setRightBorderColor(IndexedColors.BLACK.getIndex());
|
||||
blueCellStyle.setBorderTop(CellStyle.BORDER_THIN);
|
||||
blueCellStyle.setTopBorderColor(IndexedColors.BLACK.getIndex());
|
||||
blueCellStyle.setFillForegroundColor(IndexedColors.LIGHT_BLUE.getIndex());
|
||||
blueCellStyle.setFillPattern(CellStyle.SOLID_FOREGROUND);
|
||||
font.setColor(HSSFColor.WHITE.index);
|
||||
blueCellStyle.setFont(font);
|
||||
|
||||
greenCellStyle = workbook.createCellStyle();
|
||||
greenCellStyle.setBorderBottom(CellStyle.BORDER_THIN);
|
||||
greenCellStyle.setBottomBorderColor(IndexedColors.BLACK.getIndex());
|
||||
greenCellStyle.setBorderLeft(CellStyle.BORDER_THIN);
|
||||
greenCellStyle.setLeftBorderColor(IndexedColors.BLACK.getIndex());
|
||||
greenCellStyle.setBorderRight(CellStyle.BORDER_THIN);
|
||||
greenCellStyle.setRightBorderColor(IndexedColors.BLACK.getIndex());
|
||||
greenCellStyle.setBorderTop(CellStyle.BORDER_THIN);
|
||||
greenCellStyle.setTopBorderColor(IndexedColors.BLACK.getIndex());
|
||||
greenCellStyle.setFillForegroundColor(IndexedColors.GREEN.getIndex());
|
||||
greenCellStyle.setFillPattern(CellStyle.SOLID_FOREGROUND);
|
||||
font.setColor(HSSFColor.BLACK.index);
|
||||
greenCellStyle.setFont(font);
|
||||
greenCellStyle.setAlignment(CellStyle.ALIGN_CENTER);
|
||||
|
||||
yellowCellStyle = workbook.createCellStyle();
|
||||
yellowCellStyle.setBorderBottom(CellStyle.BORDER_THIN);
|
||||
yellowCellStyle.setBottomBorderColor(IndexedColors.BLACK.getIndex());
|
||||
yellowCellStyle.setBorderLeft(CellStyle.BORDER_THIN);
|
||||
yellowCellStyle.setLeftBorderColor(IndexedColors.BLACK.getIndex());
|
||||
yellowCellStyle.setBorderRight(CellStyle.BORDER_THIN);
|
||||
yellowCellStyle.setRightBorderColor(IndexedColors.BLACK.getIndex());
|
||||
yellowCellStyle.setBorderTop(CellStyle.BORDER_THIN);
|
||||
yellowCellStyle.setTopBorderColor(IndexedColors.BLACK.getIndex());
|
||||
yellowCellStyle.setFillForegroundColor(IndexedColors.YELLOW.getIndex());
|
||||
yellowCellStyle.setFillPattern(CellStyle.SOLID_FOREGROUND);
|
||||
font.setColor(HSSFColor.BLACK.index);
|
||||
yellowCellStyle.setFont(font);
|
||||
yellowCellStyle.setAlignment(CellStyle.ALIGN_CENTER);
|
||||
|
||||
orangeCellStyle = workbook.createCellStyle();
|
||||
orangeCellStyle.setBorderBottom(CellStyle.BORDER_THIN);
|
||||
orangeCellStyle.setBottomBorderColor(IndexedColors.BLACK.getIndex());
|
||||
orangeCellStyle.setBorderLeft(CellStyle.BORDER_THIN);
|
||||
orangeCellStyle.setLeftBorderColor(IndexedColors.BLACK.getIndex());
|
||||
orangeCellStyle.setBorderRight(CellStyle.BORDER_THIN);
|
||||
orangeCellStyle.setRightBorderColor(IndexedColors.BLACK.getIndex());
|
||||
orangeCellStyle.setBorderTop(CellStyle.BORDER_THIN);
|
||||
orangeCellStyle.setTopBorderColor(IndexedColors.BLACK.getIndex());
|
||||
orangeCellStyle.setFillForegroundColor(IndexedColors.ORANGE.getIndex());
|
||||
orangeCellStyle.setFillPattern(CellStyle.SOLID_FOREGROUND);
|
||||
font.setColor(HSSFColor.BLACK.index);
|
||||
orangeCellStyle.setFont(font);
|
||||
orangeCellStyle.setAlignment(CellStyle.ALIGN_CENTER);
|
||||
|
||||
redCellStyle = workbook.createCellStyle();
|
||||
redCellStyle.setBorderBottom(CellStyle.BORDER_THIN);
|
||||
redCellStyle.setBottomBorderColor(IndexedColors.BLACK.getIndex());
|
||||
redCellStyle.setBorderLeft(CellStyle.BORDER_THIN);
|
||||
redCellStyle.setLeftBorderColor(IndexedColors.BLACK.getIndex());
|
||||
redCellStyle.setBorderRight(CellStyle.BORDER_THIN);
|
||||
redCellStyle.setRightBorderColor(IndexedColors.BLACK.getIndex());
|
||||
redCellStyle.setBorderTop(CellStyle.BORDER_THIN);
|
||||
redCellStyle.setTopBorderColor(IndexedColors.BLACK.getIndex());
|
||||
redCellStyle.setFillForegroundColor(IndexedColors.RED.getIndex());
|
||||
redCellStyle.setFillPattern(CellStyle.SOLID_FOREGROUND);
|
||||
font.setColor(IndexedColors.BLACK.getIndex());
|
||||
redCellStyle.setFont(font);
|
||||
redCellStyle.setAlignment(CellStyle.ALIGN_CENTER);
|
||||
}
|
||||
|
||||
protected CellStyle getCellStyle(String type) {
|
||||
switch (type) {
|
||||
case "main":
|
||||
return mainCellStyle;
|
||||
case "left":
|
||||
return leftCellStyle;
|
||||
case "middle":
|
||||
return middleCellStyle;
|
||||
case "right":
|
||||
return rightCellStyle;
|
||||
case "blue":
|
||||
return blueCellStyle;
|
||||
case "green":
|
||||
return greenCellStyle;
|
||||
case "yellow":
|
||||
return yellowCellStyle;
|
||||
case "orange":
|
||||
return orangeCellStyle;
|
||||
case "red":
|
||||
return redCellStyle;
|
||||
default:
|
||||
return null;
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,264 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import java.io.File;
|
||||
import java.io.FileOutputStream;
|
||||
import java.io.InputStream;
|
||||
import java.io.OutputStream;
|
||||
import java.net.HttpURLConnection;
|
||||
import java.net.URL;
|
||||
import java.util.List;
|
||||
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
import android.content.pm.PackageManager;
|
||||
import android.content.pm.ResolveInfo;
|
||||
import android.net.ConnectivityManager;
|
||||
import android.net.NetworkInfo;
|
||||
import android.net.Uri;
|
||||
import android.os.AsyncTask;
|
||||
import android.os.Environment;
|
||||
import android.os.PowerManager;
|
||||
import android.webkit.MimeTypeMap;
|
||||
import android.widget.Toast;
|
||||
import cl.cromer.ubb.attendance.Progress;
|
||||
|
||||
/**
|
||||
* This class handles reading and writing to the file system.
|
||||
* @author Chris Cromer
|
||||
* Copyright 2013 -2016
|
||||
*/
|
||||
public class FileSystem {
|
||||
/**
|
||||
* This method checks the memory card to see if it is mounted and writable.
|
||||
* @return boolean Returns true if readable or false if it's not.
|
||||
*/
|
||||
public static boolean isWritable() {
|
||||
return Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState());
|
||||
}
|
||||
|
||||
/**
|
||||
* This method checks the memory card to see if it is readable.
|
||||
* @return boolean Returns true if it's readable or false if it's not.
|
||||
*/
|
||||
public static boolean isReadable() {
|
||||
return Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState()) || Environment.MEDIA_MOUNTED_READ_ONLY.equals(Environment.getExternalStorageState());
|
||||
}
|
||||
|
||||
/**
|
||||
* This method checks if a file exists on the memory card.
|
||||
* @param context The context of the calling activity.
|
||||
* @param fileName The name of the file that you want to see if it exists.
|
||||
* @return boolean Returns true if it exists or false if it doesn't.
|
||||
*/
|
||||
public static boolean fileExists(Context context, String fileName) {
|
||||
File file = new File(context.getExternalFilesDir(null), fileName);
|
||||
return file.isFile();
|
||||
}
|
||||
|
||||
/**
|
||||
* This method returns the mime type of the file based off it's extension
|
||||
* @param url The url of the file to check
|
||||
* @return String returns the mime type or null if it is invalid
|
||||
*/
|
||||
public static String getMimeType(String url) {
|
||||
String type = null;
|
||||
String extension = MimeTypeMap.getFileExtensionFromUrl(url);
|
||||
if (extension != null) {
|
||||
type = MimeTypeMap.getSingleton().getMimeTypeFromExtension(extension);
|
||||
}
|
||||
return type;
|
||||
}
|
||||
|
||||
/**
|
||||
* This method attempts to open a file for viewing/editing using a 3rd party application.
|
||||
* This method shows a toast if there isn't an application installed for the mime type.
|
||||
* @param context The context of the activity that wants to open the file.
|
||||
* @param fileName The name of the file you want to open.
|
||||
* @param mimeType The mime type of the file you want to open.
|
||||
*/
|
||||
public void openFile(Context context, String fileName, String mimeType) {
|
||||
Intent intent = new Intent();
|
||||
intent.setAction(Intent.ACTION_VIEW);
|
||||
|
||||
File file = new File(context.getExternalFilesDir(null), fileName);
|
||||
intent.setDataAndType(Uri.fromFile(file), mimeType);
|
||||
|
||||
final PackageManager packageManager = context.getPackageManager();
|
||||
List<ResolveInfo> list = packageManager.queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY);
|
||||
if (list.size() == 0) {
|
||||
Toast.makeText(context, context.getString(R.string.general_no_app), Toast.LENGTH_SHORT).show();
|
||||
}
|
||||
else {
|
||||
context.startActivity(intent);
|
||||
}
|
||||
}
|
||||
|
||||
public boolean renameFile(Context context, String oldFileName, String newFileName) {
|
||||
if (fileExists(context, oldFileName)) {
|
||||
File oldFile = new File(context.getExternalFilesDir(null), oldFileName);
|
||||
File newFile = new File(context.getExternalFilesDir(null), newFileName);
|
||||
return oldFile.renameTo(newFile);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* This method deletes the file passed
|
||||
* @param context The context of the application.
|
||||
* @param fileName The name of the file you want to delete.
|
||||
* @return boolean True on success or false if it fails.
|
||||
*/
|
||||
public boolean deleteFile(Context context, String fileName) {
|
||||
if (fileExists(context, fileName)) {
|
||||
File file = new File(context.getExternalFilesDir(null), fileName);
|
||||
return file.delete();
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* This method returns the size of the file.
|
||||
* @param context The context of the application.
|
||||
* @param fileName The name of the file you want the size of.
|
||||
* @return long The size of the file in bytes.
|
||||
*/
|
||||
public long fileSize(Context context, String fileName) {
|
||||
if (fileExists(context, fileName)) {
|
||||
File file = new File(context.getExternalFilesDir(null), fileName);
|
||||
return file.length();
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* This method returns the file path based on the version of android that is in use.
|
||||
* @param context The context of the application.
|
||||
* @param fileName The name of the file you want to get the path for.
|
||||
* @return String This string contains the full path to the file.
|
||||
*/
|
||||
public String getFilePath(Context context, String fileName) {
|
||||
return context.getExternalFilesDir(null) + "/" + fileName;
|
||||
}
|
||||
|
||||
/**
|
||||
* This class is an AsycnTask used to download a file to the memory card.
|
||||
* The construct requires 2 fields. Context and Filename.
|
||||
* In the execute command; a URL is needed to download.
|
||||
* This class will call a listener if you pass one through downloadListener.
|
||||
* @author cromer
|
||||
*/
|
||||
public class Download extends AsyncTask<String, Integer, Boolean> {
|
||||
private Context context;
|
||||
private PowerManager.WakeLock wakeLock;
|
||||
private String fileName;
|
||||
public Progress progress = null; // If this is set the progress bar will be updated
|
||||
public ListenerDownload downloadListener = null; // This needs to be set from the parent activity
|
||||
|
||||
/**
|
||||
* This construct is used to configure the class to run.
|
||||
* @param context The context of the parent activity.
|
||||
* @param fileName The name of the file you want to save to the memory card.
|
||||
*/
|
||||
public Download(Context context, String fileName) {
|
||||
this.context = context;
|
||||
this.fileName = fileName;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
super.onPreExecute();
|
||||
// Let's make sure the CPU doesn't go to sleep
|
||||
PowerManager powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, getClass().getName());
|
||||
if (!wakeLock.isHeld()) {
|
||||
wakeLock.acquire();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Boolean doInBackground(String... receivedUrl) {
|
||||
InputStream inStream;
|
||||
OutputStream outStream;
|
||||
|
||||
ConnectivityManager cm = (ConnectivityManager) context.getSystemService(Context.CONNECTIVITY_SERVICE);
|
||||
NetworkInfo activeNetwork = cm.getActiveNetworkInfo();
|
||||
|
||||
if (activeNetwork == null) {
|
||||
// Couldn't get a connection
|
||||
return false;
|
||||
}
|
||||
|
||||
if (!activeNetwork.isConnected()) {
|
||||
// Not connected
|
||||
return false;
|
||||
}
|
||||
|
||||
try {
|
||||
URL url = new URL(receivedUrl[0]);
|
||||
HttpURLConnection connection = (HttpURLConnection) url.openConnection();
|
||||
connection.setReadTimeout(10000);
|
||||
connection.setConnectTimeout(15000);
|
||||
connection.connect();
|
||||
|
||||
// Problem with the server and file
|
||||
if (connection.getResponseCode() != HttpURLConnection.HTTP_OK) {
|
||||
return false;
|
||||
}
|
||||
|
||||
int fileLength = connection.getContentLength();
|
||||
|
||||
inStream = connection.getInputStream();
|
||||
|
||||
File file = new File(context.getExternalFilesDir(null), fileName);
|
||||
outStream = new FileOutputStream(file);
|
||||
|
||||
byte data[] = new byte[4096];
|
||||
long total = 0;
|
||||
int count;
|
||||
while ((count = inStream.read(data)) != -1) {
|
||||
// They cancelled the download
|
||||
if (isCancelled()) {
|
||||
inStream.close();
|
||||
outStream.close();
|
||||
return file.delete(); // If they cancelled the download, delete what was downloaded
|
||||
}
|
||||
total += count;
|
||||
|
||||
if (fileLength > 0) {
|
||||
// Update the progress meter if we know how large the file is
|
||||
publishProgress((int) (total * 100 / fileLength));
|
||||
}
|
||||
outStream.write(data, 0, count);
|
||||
}
|
||||
outStream.close();
|
||||
inStream.close();
|
||||
connection.disconnect();
|
||||
return true;
|
||||
}
|
||||
catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onProgressUpdate(Integer... progressUpdate) {
|
||||
super.onProgressUpdate(progressUpdate);
|
||||
// if we get here, length is known, now set indeterminate to false and show how long till it's finished
|
||||
if (progress != null) {
|
||||
progress.setIndeterminate(false);
|
||||
progress.setProgress(progressUpdate[0]);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Boolean result) {
|
||||
wakeLock.release();
|
||||
if (downloadListener != null) {
|
||||
// Call the listener if one was set
|
||||
downloadListener.onDownloadComplete(result);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,751 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.content.ContentResolver;
|
||||
import android.content.ContentValues;
|
||||
import android.content.Context;
|
||||
import android.content.DialogInterface;
|
||||
import android.content.Intent;
|
||||
import android.database.Cursor;
|
||||
import android.database.sqlite.SQLiteDatabase;
|
||||
import android.net.Uri;
|
||||
import android.os.AsyncTask;
|
||||
import android.os.Handler;
|
||||
import android.os.PowerManager;
|
||||
import android.support.v7.app.AppCompatActivity;
|
||||
import android.os.Bundle;
|
||||
import android.util.Log;
|
||||
import android.widget.Toast;
|
||||
|
||||
import org.apache.poi.hssf.usermodel.HSSFWorkbook;
|
||||
import org.apache.poi.poifs.filesystem.POIFSFileSystem;
|
||||
import org.apache.poi.ss.usermodel.Cell;
|
||||
import org.apache.poi.ss.usermodel.Row;
|
||||
import org.apache.poi.ss.usermodel.Sheet;
|
||||
import org.apache.poi.ss.usermodel.Workbook;
|
||||
|
||||
import java.io.FileInputStream;
|
||||
import java.io.FileNotFoundException;
|
||||
import java.io.IOException;
|
||||
import java.io.InputStream;
|
||||
import java.util.regex.Matcher;
|
||||
import java.util.regex.Pattern;
|
||||
|
||||
import cl.cromer.ubb.attendance.Progress;
|
||||
import cl.cromer.ubb.attendance.RUT;
|
||||
import cl.cromer.ubb.attendance.DBSchema.DBSubjects;
|
||||
import cl.cromer.ubb.attendance.DBSchema.DBMajors;
|
||||
import cl.cromer.ubb.attendance.DBSchema.DBStudents;
|
||||
import cl.cromer.ubb.attendance.DBSchema.DBCourses;
|
||||
import cl.cromer.ubb.attendance.DBSchema.DBCoursesStudents;
|
||||
|
||||
public class ImportExcel extends AppCompatActivity {
|
||||
|
||||
// SQLite database
|
||||
private SQLParser sqlParser = null;
|
||||
private SQLiteDatabase ubbDB = null;
|
||||
|
||||
// Background thread for the database
|
||||
private Thread thread = null;
|
||||
private Handler threadHandler = new Handler();
|
||||
|
||||
// Progress bar
|
||||
private Progress progress = null;
|
||||
|
||||
// Background async
|
||||
private ReadFile readFile = null;
|
||||
|
||||
@Override
|
||||
protected void onCreate(Bundle savedInstanceState) {
|
||||
super.onCreate(savedInstanceState);
|
||||
setContentView(R.layout.activity_import_excel);
|
||||
|
||||
// Receive the info about the excel file
|
||||
Intent subjectListIntent = getIntent();
|
||||
final int type = subjectListIntent.getIntExtra(StaticVariables.IMPORT_EXCEL_FILE_TYPE, 1);
|
||||
final String path = subjectListIntent.getStringExtra(StaticVariables.IMPORT_EXCEL_FILE);
|
||||
|
||||
// Create a progress dialog for slow devices
|
||||
progress = new Progress();
|
||||
progress.show(this, 1);
|
||||
progress.setCancelable(false);
|
||||
|
||||
// Load the SQLite database
|
||||
sqlParser = new SQLParser(this);
|
||||
thread = new Thread(new Runnable() {
|
||||
public void run() {
|
||||
ubbDB = sqlParser.getWritableDatabase();
|
||||
threadHandler.post(new Runnable() {
|
||||
public void run() {
|
||||
databaseLoaded(type, path);
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
thread.start();
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
thread.interrupt();
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onDestroy() {
|
||||
super.onDestroy();
|
||||
// We need to get rid of the progressbar if it's showing
|
||||
if (progress != null && progress.isShowing()) {
|
||||
progress.dismiss();
|
||||
}
|
||||
// Kill the other background thread before we start the new thread
|
||||
if (readFile != null) {
|
||||
readFile.cancel(true);
|
||||
}
|
||||
}
|
||||
|
||||
private void databaseLoaded(int type, String path) {
|
||||
// The database has finished loading, so read the file
|
||||
readFile = new ReadFile(getApplicationContext(), type, path);
|
||||
readFile.execute();
|
||||
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
readFile.cancel(true);
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private class ReadFile extends AsyncTask<Void, Void, Subject> {
|
||||
private Context context;
|
||||
private PowerManager.WakeLock wakeLock;
|
||||
|
||||
private int type;
|
||||
private String path;
|
||||
|
||||
protected ReadFile(Context context, int type, String path) {
|
||||
this.context = context;
|
||||
this.type = type;
|
||||
this.path = path;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
super.onPreExecute();
|
||||
// Let's make sure the CPU doesn't go to sleep
|
||||
PowerManager powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, getClass().getName());
|
||||
if (!wakeLock.isHeld()) {
|
||||
wakeLock.acquire();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Subject doInBackground(Void... voids) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
Subject subject = new Subject();
|
||||
|
||||
InputStream inputStream = null;
|
||||
if (type == 0) {
|
||||
ContentResolver contentResolver = getContentResolver();
|
||||
try {
|
||||
inputStream = contentResolver.openInputStream(Uri.parse(path));
|
||||
}
|
||||
catch (FileNotFoundException e) {
|
||||
if (BuildConfig.DEBUG) {
|
||||
e.printStackTrace();
|
||||
Toast.makeText(context, R.string.import_not_valid_excel, Toast.LENGTH_SHORT).show();
|
||||
Log.d("ImportExcel", e.getMessage());
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
}
|
||||
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
Workbook workbook = null;
|
||||
|
||||
try {
|
||||
POIFSFileSystem fs;
|
||||
if (inputStream != null) {
|
||||
fs = new POIFSFileSystem(inputStream);
|
||||
inputStream.close();
|
||||
}
|
||||
else {
|
||||
fs = new POIFSFileSystem(new FileInputStream(path));
|
||||
}
|
||||
workbook = new HSSFWorkbook(fs);
|
||||
|
||||
}
|
||||
catch (IOException e) {
|
||||
if (BuildConfig.DEBUG) {
|
||||
e.printStackTrace();
|
||||
Log.d("ImportExcel", e.getMessage());
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
|
||||
Sheet sheet = null;
|
||||
if (workbook != null) {
|
||||
// Make it so it returns blank when the cell does not exist
|
||||
workbook.setMissingCellPolicy(Row.MissingCellPolicy.CREATE_NULL_AS_BLANK);
|
||||
sheet = workbook.getSheetAt(0);
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
|
||||
Pattern pattern;
|
||||
Matcher matcher = null;
|
||||
Row row = null;
|
||||
Cell cell = null;
|
||||
String cellContentString = null;
|
||||
int cellContentInt = 0;
|
||||
|
||||
// Get the major
|
||||
if (sheet != null) {
|
||||
row = sheet.getRow(5);
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
if (row != null) {
|
||||
cell = row.getCell(7);
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
if (cell != null) {
|
||||
cellContentString = cell.getRichStringCellValue().getString();
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
|
||||
// Let's play the matching game
|
||||
pattern = Pattern.compile("\\(([0-9]+)\\)\\s(.+)");
|
||||
if (cellContentString != null) {
|
||||
matcher = pattern.matcher(cellContentString);
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
|
||||
if (matcher != null) {
|
||||
if (matcher.find()) {
|
||||
subject.setMajorCode(Integer.valueOf(matcher.group(1)));
|
||||
subject.setMajorName(StringFixer.fixCase(matcher.group(2)));
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
// My favorite debug logger!
|
||||
Log.wtf("ImportExcel", "Really? Did you really mess with the excel file...?");
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
}
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
// Check for major
|
||||
Cursor cursor = ubbDB.query(
|
||||
DBMajors.TABLE_NAME,
|
||||
new String[] {DBMajors.COLUMN_ID},
|
||||
DBMajors.COLUMN_CODE + "=" + subject.getMajorCode(),
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
"1");
|
||||
|
||||
ContentValues values = new ContentValues();
|
||||
values.put(DBMajors.COLUMN_CODE, subject.getMajorCode());
|
||||
values.put(DBMajors.COLUMN_NAME, subject.getMajorName());
|
||||
|
||||
if (cursor.getCount() > 0) {
|
||||
// It already exists, so let's update it
|
||||
cursor.moveToFirst();
|
||||
subject.setMajorId(cursor.getInt(cursor.getColumnIndex(DBMajors.COLUMN_ID)));
|
||||
ubbDB.update(DBMajors.TABLE_NAME, values, DBMajors.COLUMN_CODE + "=" + subject.getMajorCode(), null);
|
||||
}
|
||||
else {
|
||||
// No, it does not exist, let's make it
|
||||
subject.setMajorId((int) ubbDB.insert(DBMajors.TABLE_NAME, null, values));
|
||||
}
|
||||
cursor.close();
|
||||
|
||||
// Get the subject and code
|
||||
if (sheet != null) {
|
||||
row = sheet.getRow(1);
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
if (row != null) {
|
||||
cell = row.getCell(2);
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
if (cell != null) {
|
||||
cellContentString = cell.getRichStringCellValue().getString();
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
|
||||
// Match the code and name using regex
|
||||
pattern = Pattern.compile("\\(([0-9]+)\\)\\s(.+)");
|
||||
if (cellContentString != null) {
|
||||
matcher = pattern.matcher(cellContentString);
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
if (matcher != null) {
|
||||
if (matcher.find()) {
|
||||
subject.setSubjectCode(Integer.valueOf(matcher.group(1)));
|
||||
subject.setSubjectName(StringFixer.fixCase(matcher.group(2)));
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
// My favorite debug logger!
|
||||
Log.wtf("ImportExcel", "Really? Did you really mess with the excel file...?");
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
}
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
|
||||
// Check if the subject already exists
|
||||
cursor = ubbDB.query(
|
||||
DBSubjects.TABLE_NAME,
|
||||
new String[]{DBSubjects.COLUMN_ID},
|
||||
DBSubjects.COLUMN_CODE + "=" + subject.getSubjectCode(),
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
"1");
|
||||
|
||||
values = new ContentValues();
|
||||
values.put(DBSubjects.COLUMN_CODE, subject.getSubjectCode());
|
||||
values.put(DBSubjects.COLUMN_NAME, subject.getSubjectName());
|
||||
values.put(DBSubjects.COLUMN_MAJOR, subject.getMajorId());
|
||||
|
||||
if (cursor.getCount() > 0) {
|
||||
// It already exists, so let's update it
|
||||
cursor.moveToFirst();
|
||||
subject.setSubjectId(cursor.getInt(cursor.getColumnIndex(DBSubjects.COLUMN_ID)));
|
||||
ubbDB.update(DBSubjects.TABLE_NAME, values, DBSubjects.COLUMN_CODE + "=" + subject.getSubjectCode(), null);
|
||||
}
|
||||
else {
|
||||
// No, it does not exist, let's make it
|
||||
subject.setSubjectId((int) ubbDB.insert(DBSubjects.TABLE_NAME, null, values));
|
||||
}
|
||||
cursor.close();
|
||||
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
Course course = new Course(subject);
|
||||
|
||||
// Get the course's year and semester
|
||||
if (sheet != null) {
|
||||
row = sheet.getRow(2);
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
if (row != null) {
|
||||
cell = row.getCell(3);
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
if (cell != null) {
|
||||
cellContentString = cell.getRichStringCellValue().getString();
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
|
||||
// Find the values!
|
||||
pattern = Pattern.compile("([0-9]+)\\s-\\s([1-2])");
|
||||
if (cellContentString != null) {
|
||||
matcher = pattern.matcher(cellContentString);
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
if (matcher != null) {
|
||||
if (matcher.find()) {
|
||||
course.setCourseYear(Integer.valueOf(matcher.group(1)));
|
||||
course.setCourseSemester(Integer.valueOf(matcher.group(2)));
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
// My favorite debug logger!
|
||||
Log.wtf("ImportExcel", "Really? Did you really mess with the excel file...?");
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
}
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
// Get the section
|
||||
if (sheet != null) {
|
||||
row = sheet.getRow(5);
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
if (row != null) {
|
||||
cell = row.getCell(5);
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
if (cell != null) {
|
||||
cellContentInt = (int) cell.getNumericCellValue();
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
|
||||
course.setCourseSection(cellContentInt);
|
||||
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
// Check if the course already exists
|
||||
cursor = ubbDB.query(
|
||||
DBCourses.TABLE_NAME,
|
||||
new String[] {DBCourses.COLUMN_ID},
|
||||
DBCourses.COLUMN_SUBJECT + "=" + course.getSubjectId() + " AND " +
|
||||
DBCourses.COLUMN_YEAR + "=" + course.getYear() + " AND " +
|
||||
DBCourses.COLUMN_SEMESTER + "=" + course.getCourseSemester() + " AND " +
|
||||
DBCourses.COLUMN_SECTION + "=" + course.getCourseSection(),
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
"1");
|
||||
|
||||
if (cursor.getCount() == 0) {
|
||||
// Doesn't exist, so let's make it
|
||||
values = new ContentValues();
|
||||
values.put(DBCourses.COLUMN_SUBJECT, course.getSubjectId());
|
||||
values.put(DBCourses.COLUMN_YEAR, course.getYear());
|
||||
values.put(DBCourses.COLUMN_SEMESTER, course.getCourseSemester());
|
||||
values.put(DBCourses.COLUMN_SECTION, course.getCourseSection());
|
||||
course.setCourseId((int) ubbDB.insert(DBCourses.TABLE_NAME, null, values));
|
||||
}
|
||||
else {
|
||||
cursor.moveToFirst();
|
||||
course.setCourseId(cursor.getInt(cursor.getColumnIndex(DBCourses.COLUMN_ID)));
|
||||
}
|
||||
|
||||
cursor.close();
|
||||
|
||||
// Get number of students
|
||||
if (sheet != null) {
|
||||
row = sheet.getRow(3);
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
// My favorite debug logger!
|
||||
Log.wtf("ImportExcel", "Really? Did you really mess with the excel file...?");
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
}
|
||||
if (row != null) {
|
||||
cell = row.getCell(2);
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
// My favorite debug logger!
|
||||
Log.wtf("ImportExcel", "Really? Did you really mess with the excel file...?");
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
}
|
||||
if (cell != null) {
|
||||
cellContentString = cell.getRichStringCellValue().getString();
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
// My favorite debug logger!
|
||||
Log.wtf("ImportExcel", "Really? Did you really mess with the excel file...?");
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
}
|
||||
|
||||
int numberOfStudents = 0;
|
||||
// Match the code and name using regex
|
||||
pattern = Pattern.compile("([0-9]+)");
|
||||
if (cellContentString != null) {
|
||||
matcher = pattern.matcher(cellContentString);
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
if (matcher != null) {
|
||||
if (matcher.find()) {
|
||||
numberOfStudents = Integer.valueOf(matcher.group(1));
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
// My favorite debug logger!
|
||||
Log.wtf("ImportExcel", "Really? Did you really mess with the excel file...?");
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
}
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "How the hell did that?");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
// Get students
|
||||
for (int i = 5; i < numberOfStudents + 5; i++) {
|
||||
Student student = new Student();
|
||||
assert sheet != null;
|
||||
row = sheet.getRow(i);
|
||||
|
||||
// Get RUN
|
||||
cell = row.getCell(1);
|
||||
cellContentString = RUT.cleanRut(StringFixer.fixCase(cell.getRichStringCellValue().getString()));
|
||||
if (!RUT.isValidRut(cellContentString)) {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportExcel", "The run \"" + cellContentString + "\" is invalid!");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
student.setRun(cellContentString);
|
||||
|
||||
// Get names
|
||||
cell = row.getCell(2);
|
||||
cellContentString = StringFixer.fixCase(cell.getRichStringCellValue().getString());
|
||||
String[] names = cellContentString.split("\\s", 2);
|
||||
|
||||
student.setFirstName(names[0]);
|
||||
student.setSecondName(names[1]);
|
||||
|
||||
// Get last names
|
||||
cell = row.getCell(3);
|
||||
cellContentString = StringFixer.fixCase(cell.getRichStringCellValue().getString());
|
||||
student.setFirstLastName(cellContentString);
|
||||
cell = row.getCell(4);
|
||||
if (cell.getCellType() == Cell.CELL_TYPE_BLANK) {
|
||||
// Oh my god, it's a gringo!
|
||||
student.setSecondLastName("");
|
||||
}
|
||||
else {
|
||||
cellContentString = StringFixer.fixCase(cell.getRichStringCellValue().getString());
|
||||
student.setSecondLastName(cellContentString);
|
||||
}
|
||||
|
||||
// Major id
|
||||
student.setMajor(subject.getMajorId());
|
||||
|
||||
// Entered the uni in what year?
|
||||
cell = row.getCell(8);
|
||||
cellContentInt = (int) cell.getNumericCellValue();
|
||||
student.setEnrolled(cellContentInt);
|
||||
|
||||
// Email
|
||||
cell = row.getCell(9);
|
||||
cellContentString = StringFixer.fixCase(cell.getRichStringCellValue().getString());
|
||||
student.setEmail(cellContentString);
|
||||
|
||||
// Check if student exists in the database
|
||||
cursor = ubbDB.query(
|
||||
DBStudents.TABLE_NAME,
|
||||
new String[] {DBStudents.COLUMN_ID},
|
||||
DBStudents.COLUMN_RUN + "=" + student.getRun(),
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
"1");
|
||||
|
||||
values = new ContentValues();
|
||||
values.put(DBStudents.COLUMN_RUN, student.getRun());
|
||||
values.put(DBStudents.COLUMN_FIRST_NAME, student.getFirstName());
|
||||
values.put(DBStudents.COLUMN_SECOND_NAME, student.getSecondName());
|
||||
values.put(DBStudents.COLUMN_FIRST_LAST_NAME, student.getFirstLastName());
|
||||
values.put(DBStudents.COLUMN_SECOND_LAST_NAME, student.getSecondLastName());
|
||||
values.put(DBStudents.COLUMN_MAJOR, student.getMajor());
|
||||
values.put(DBStudents.COLUMN_ENROLLED, student.getEnrolled());
|
||||
values.put(DBStudents.COLUMN_EMAIL, student.getEmail());
|
||||
|
||||
if (cursor.getCount() > 0) {
|
||||
// They already exist, update their info
|
||||
cursor.moveToFirst();
|
||||
student.setStudentId(cursor.getInt(cursor.getColumnIndex(DBStudents.COLUMN_ID)));
|
||||
ubbDB.update(DBStudents.TABLE_NAME, values, DBStudents.COLUMN_RUN + "=" + student.getRun(), null);
|
||||
}
|
||||
else {
|
||||
// No, they does not exist, let's make them
|
||||
student.setStudentId((int) ubbDB.insert(DBStudents.TABLE_NAME, null, values));
|
||||
}
|
||||
cursor.close();
|
||||
|
||||
// Check if the student belongs to the course, if not add them
|
||||
cursor = ubbDB.query(
|
||||
DBCoursesStudents.TABLE_NAME,
|
||||
new String[] {DBCoursesStudents.COLUMN_ID},
|
||||
DBCoursesStudents.COLUMN_STUDENT + "=" + student.getStudentId() + " AND " + DBCoursesStudents.COLUMN_COURSE + "=" + course.getCourseId(),
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
"1");
|
||||
|
||||
if (cursor.getCount() == 0) {
|
||||
// No, they does not exist, let's make them
|
||||
values = new ContentValues();
|
||||
values.put(DBCoursesStudents.COLUMN_STUDENT, student.getStudentId());
|
||||
values.put(DBCoursesStudents.COLUMN_COURSE, course.getCourseId());
|
||||
ubbDB.insert(DBCoursesStudents.TABLE_NAME, null, values);
|
||||
}
|
||||
cursor.close();
|
||||
}
|
||||
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
|
||||
return subject;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Subject subject) {
|
||||
// Release the kraken errr wakelock
|
||||
progress.dismiss();
|
||||
wakeLock.release();
|
||||
Intent intent = new Intent();
|
||||
intent.putExtra(StaticVariables.SUBJECT_OBJECT, subject);
|
||||
setResult(RESULT_OK, intent);
|
||||
finish();
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,411 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.app.ActivityManager;
|
||||
import android.content.ContentResolver;
|
||||
import android.content.ContentValues;
|
||||
import android.content.Context;
|
||||
import android.content.DialogInterface;
|
||||
import android.content.Intent;
|
||||
import android.database.Cursor;
|
||||
import android.database.sqlite.SQLiteDatabase;
|
||||
import android.graphics.Bitmap;
|
||||
import android.graphics.BitmapFactory;
|
||||
import android.net.Uri;
|
||||
import android.os.AsyncTask;
|
||||
import android.os.Handler;
|
||||
import android.os.PowerManager;
|
||||
import android.support.v7.app.AppCompatActivity;
|
||||
import android.os.Bundle;
|
||||
import android.util.Log;
|
||||
import android.widget.Toast;
|
||||
|
||||
import java.io.ByteArrayOutputStream;
|
||||
import java.io.File;
|
||||
import java.io.FileInputStream;
|
||||
import java.io.FileNotFoundException;
|
||||
import java.io.IOException;
|
||||
import java.io.InputStream;
|
||||
|
||||
import cl.cromer.ubb.attendance.Progress;
|
||||
import cl.cromer.ubb.attendance.RUT;
|
||||
import cl.cromer.ubb.attendance.DBSchema.DBStudents;
|
||||
|
||||
public class ImportPhotos extends AppCompatActivity {
|
||||
|
||||
// SQLite database
|
||||
private SQLParser sqlParser = null;
|
||||
private SQLiteDatabase ubbDB = null;
|
||||
|
||||
// Background thread for the database
|
||||
private Thread thread = null;
|
||||
private Handler threadHandler = new Handler();
|
||||
|
||||
// Progress bar
|
||||
private Progress progress = null;
|
||||
|
||||
// Background async
|
||||
private ReadFile readFile = null;
|
||||
|
||||
@Override
|
||||
protected void onCreate(Bundle savedInstanceState) {
|
||||
super.onCreate(savedInstanceState);
|
||||
setContentView(R.layout.activity_import_photos);
|
||||
|
||||
// Receive the info about the photo zip
|
||||
Intent subjectListIntent = getIntent();
|
||||
final int type = subjectListIntent.getIntExtra(StaticVariables.IMPORT_PHOTO_ZIP_FILE_TYPE, 1);
|
||||
final String path = subjectListIntent.getStringExtra(StaticVariables.IMPORT_PHOTO_ZIP_FILE);
|
||||
|
||||
// Create a progress dialog for slow devices
|
||||
progress = new Progress();
|
||||
progress.show(this, 1);
|
||||
progress.setCancelable(false);
|
||||
|
||||
// Load the SQLite database
|
||||
sqlParser = new SQLParser(this);
|
||||
thread = new Thread(new Runnable() {
|
||||
public void run() {
|
||||
ubbDB = sqlParser.getWritableDatabase();
|
||||
threadHandler.post(new Runnable() {
|
||||
public void run() {
|
||||
databaseLoaded(type, path);
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
thread.start();
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
thread.interrupt();
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onDestroy() {
|
||||
super.onDestroy();
|
||||
// We need to get rid of the progressbar if it's showing
|
||||
if (progress != null && progress.isShowing()) {
|
||||
progress.dismiss();
|
||||
}
|
||||
// Kill the other background thread before we start the new thread
|
||||
if (readFile != null) {
|
||||
readFile.cancel(true);
|
||||
}
|
||||
}
|
||||
|
||||
private void databaseLoaded(int type, String path) {
|
||||
// The database has finished loading, so read the file
|
||||
readFile = new ReadFile(getApplicationContext(), type, path);
|
||||
readFile.execute();
|
||||
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
readFile.cancel(true);
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private class ReadFile extends AsyncTask<Void, Void, Void> {
|
||||
private Context context;
|
||||
private PowerManager.WakeLock wakeLock;
|
||||
|
||||
private int type;
|
||||
private String path;
|
||||
|
||||
protected ReadFile(Context context, int type, String path) {
|
||||
this.context = context;
|
||||
this.type = type;
|
||||
this.path = path;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
super.onPreExecute();
|
||||
// Let's make sure the CPU doesn't go to sleep
|
||||
PowerManager powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, getClass().getName());
|
||||
if (!wakeLock.isHeld()) {
|
||||
wakeLock.acquire();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Void doInBackground(Void... voids) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
InputStream inputStream = null;
|
||||
if (type == 0) {
|
||||
ContentResolver contentResolver = getContentResolver();
|
||||
try {
|
||||
inputStream = contentResolver.openInputStream(Uri.parse(path));
|
||||
}
|
||||
catch (FileNotFoundException e) {
|
||||
if (BuildConfig.DEBUG) {
|
||||
e.printStackTrace();
|
||||
Toast.makeText(context, R.string.import_not_valid_zip, Toast.LENGTH_SHORT).show();
|
||||
Log.d("ImportPhotos", e.getMessage());
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
}
|
||||
|
||||
//File directory = getExternalFilesDir(null);
|
||||
File directory = getFilesDir();
|
||||
|
||||
ZipFile zipFile = new ZipFile();
|
||||
if (inputStream != null) {
|
||||
try {
|
||||
if (!zipFile.unzip(inputStream, directory)) {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Toast.makeText(context, R.string.import_not_valid_zip, Toast.LENGTH_SHORT).show();
|
||||
Log.wtf("ImportPhotos", "Something strange happened in ZipFile.");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
inputStream.close();
|
||||
}
|
||||
catch (IOException e) {
|
||||
if (BuildConfig.DEBUG) {
|
||||
e.printStackTrace();
|
||||
Toast.makeText(context, R.string.import_not_valid_zip, Toast.LENGTH_SHORT).show();
|
||||
Log.d("ImportPhotos", e.getMessage());
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
}
|
||||
else {
|
||||
try {
|
||||
if (!zipFile.unzip(new FileInputStream(path), directory)) {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Toast.makeText(context, R.string.import_not_valid_zip, Toast.LENGTH_SHORT).show();
|
||||
Log.wtf("ImportPhotos", "Something strange happened in ZipFile.");
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
}
|
||||
catch (IOException e) {
|
||||
if (BuildConfig.DEBUG) {
|
||||
e.printStackTrace();
|
||||
Toast.makeText(context, R.string.import_not_valid_zip, Toast.LENGTH_SHORT).show();
|
||||
Log.d("ImportPhotos", e.getMessage());
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
}
|
||||
|
||||
// Update all the name columns
|
||||
normalizeNameColumns(false);
|
||||
|
||||
for (File file : directory.listFiles()) {
|
||||
String[] temp = file.toString().split("/");
|
||||
String fileName = temp[temp.length - 1];
|
||||
|
||||
String run = null;
|
||||
if (RUT.isValidRut(RUT.cleanRut(fileName.substring(0, fileName.length() - 4)))) {
|
||||
run = RUT.cleanRut(fileName.substring(0, fileName.length() - 4));
|
||||
}
|
||||
|
||||
if (run == null) {
|
||||
// Search by name
|
||||
String nameTemp = StringFixer.normalizer(fileName.substring(0, fileName.length() - 4));
|
||||
String[] names = nameTemp.split("(?=\\p{Lu})");
|
||||
|
||||
String where;
|
||||
if (names.length < 3) {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportPhotos", "The file name is wrong");
|
||||
}
|
||||
continue;
|
||||
}
|
||||
else if (names.length == 3) {
|
||||
// First name, last name
|
||||
where = DBStudents.COLUMN_FIRST_NAME_NORM + " LIKE \"" + names[1] + "\" AND " + DBStudents.COLUMN_FIRST_LAST_NAME_NORM + " LIKE \"" + names[2] + "\"";
|
||||
}
|
||||
else if (names.length == 4) {
|
||||
// First name, last names
|
||||
where = DBStudents.COLUMN_FIRST_NAME_NORM + " LIKE \"" + names[1] + "\" AND " + DBStudents.COLUMN_FIRST_LAST_NAME_NORM + " LIKE \"" + names[2] + "\" AND " + DBStudents.COLUMN_SECOND_LAST_NAME_NORM + " LIKE \"" + names[3] + "\"";
|
||||
}
|
||||
else {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportPhotos", "The file name is wrong");
|
||||
}
|
||||
continue;
|
||||
}
|
||||
|
||||
Cursor cursor = ubbDB.query(
|
||||
DBStudents.TABLE_NAME,
|
||||
new String[] {DBStudents.COLUMN_ID},
|
||||
where,
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
"1");
|
||||
|
||||
if (cursor.getCount() > 0) {
|
||||
// Found a student, let's load his image
|
||||
BitmapFactory.Options options = new BitmapFactory.Options();
|
||||
ActivityManager activityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
|
||||
if (activityManager.getMemoryClass() <= 32) {
|
||||
options.inSampleSize = 2; // Shrink quality
|
||||
}
|
||||
else {
|
||||
options.inSampleSize = 1; // Full quality
|
||||
}
|
||||
options.inPreferredConfig = Bitmap.Config.ARGB_8888;
|
||||
FileInputStream fileInputStream = null;
|
||||
try {
|
||||
fileInputStream = new FileInputStream(file);
|
||||
}
|
||||
catch (FileNotFoundException e) {
|
||||
if (BuildConfig.DEBUG) {
|
||||
e.printStackTrace();
|
||||
Toast.makeText(context, R.string.import_not_valid_zip, Toast.LENGTH_SHORT).show();
|
||||
Log.d("ImportPhotos", e.getMessage());
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
Bitmap bitmap = BitmapFactory.decodeStream(fileInputStream, null, options);
|
||||
byte[] imgByte = bitmapToByte(bitmap);
|
||||
bitmap.recycle();
|
||||
|
||||
ContentValues values = new ContentValues();
|
||||
values.put(DBStudents.COLUMN_PHOTO, imgByte);
|
||||
ubbDB.update(DBStudents.TABLE_NAME, values, where, null);
|
||||
}
|
||||
cursor.close();
|
||||
}
|
||||
else {
|
||||
// Search by run
|
||||
Cursor cursor = ubbDB.query(
|
||||
DBStudents.TABLE_NAME,
|
||||
new String[] {DBStudents.COLUMN_ID},
|
||||
DBStudents.COLUMN_RUN + "=\"" + run + "\"",
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
"1");
|
||||
|
||||
if (cursor.getCount() > 0) {
|
||||
// Found a student, let's load his image
|
||||
BitmapFactory.Options options = new BitmapFactory.Options();
|
||||
ActivityManager activityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
|
||||
if (activityManager.getMemoryClass() <= 32) {
|
||||
options.inSampleSize = 2; // Shrink quality
|
||||
}
|
||||
else {
|
||||
options.inSampleSize = 1; // Full quality
|
||||
}
|
||||
options.inPreferredConfig = Bitmap.Config.ARGB_8888;
|
||||
FileInputStream fileInputStream = null;
|
||||
try {
|
||||
fileInputStream = new FileInputStream(file);
|
||||
}
|
||||
catch (FileNotFoundException e) {
|
||||
if (BuildConfig.DEBUG) {
|
||||
e.printStackTrace();
|
||||
Toast.makeText(context, R.string.import_not_valid_zip, Toast.LENGTH_SHORT).show();
|
||||
Log.d("ImportPhotos", e.getMessage());
|
||||
}
|
||||
setResult(RESULT_CANCELED);
|
||||
finish();
|
||||
}
|
||||
Bitmap bitmap = BitmapFactory.decodeStream(fileInputStream, null, options);
|
||||
byte[] imgByte = bitmapToByte(bitmap);
|
||||
bitmap.recycle();
|
||||
|
||||
ContentValues values = new ContentValues();
|
||||
values.put(DBStudents.COLUMN_PHOTO, imgByte);
|
||||
ubbDB.update(DBStudents.TABLE_NAME, values, DBStudents.COLUMN_RUN + "=" + run, null);
|
||||
}
|
||||
cursor.close();
|
||||
}
|
||||
|
||||
if (file.exists()) {
|
||||
if (!file.delete()) {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportPhotos", "The file couldn't be deleted?");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (!directory.delete()) {
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.wtf("ImportPhotos", "The directory could not be deleted?");
|
||||
}
|
||||
}
|
||||
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Void voids) {
|
||||
// Release the kraken errr wakelock
|
||||
progress.dismiss();
|
||||
wakeLock.release();
|
||||
Intent intent = new Intent();
|
||||
setResult(RESULT_OK, intent);
|
||||
finish();
|
||||
}
|
||||
}
|
||||
|
||||
private byte[] bitmapToByte(Bitmap bitmap) {
|
||||
ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
|
||||
bitmap.compress(Bitmap.CompressFormat.PNG, 0, outputStream);
|
||||
return outputStream.toByteArray();
|
||||
}
|
||||
|
||||
// This function is necessary because SQLite in android can't ignore accented characters...
|
||||
private void normalizeNameColumns(boolean redo) {
|
||||
String redoString = null;
|
||||
if (!redo) {
|
||||
redoString = DBStudents.COLUMN_FIRST_NAME_NORM + " IS NULL OR " + DBStudents.COLUMN_FIRST_NAME_NORM + "=\"\"";
|
||||
}
|
||||
|
||||
Cursor cursor = ubbDB.query(
|
||||
DBStudents.TABLE_NAME,
|
||||
DBStudents.ALL_COLUMNS,
|
||||
redoString,
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
null);
|
||||
|
||||
if (cursor.getCount() > 0) {
|
||||
while (cursor.moveToNext()) {
|
||||
int id = cursor.getInt(cursor.getColumnIndex(DBStudents.COLUMN_ID));
|
||||
String firstName = StringFixer.normalizer(cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_FIRST_NAME)));
|
||||
String secondName = StringFixer.normalizer(cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_SECOND_NAME)));
|
||||
String firstLastName = StringFixer.normalizer(cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_FIRST_LAST_NAME)));
|
||||
String secondLastName = StringFixer.normalizer(cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_SECOND_LAST_NAME)));
|
||||
|
||||
ContentValues values = new ContentValues();
|
||||
values.put(DBStudents.COLUMN_FIRST_NAME_NORM, firstName);
|
||||
values.put(DBStudents.COLUMN_SECOND_NAME_NORM, secondName);
|
||||
values.put(DBStudents.COLUMN_FIRST_LAST_NAME_NORM, firstLastName);
|
||||
values.put(DBStudents.COLUMN_SECOND_LAST_NAME_NORM, secondLastName);
|
||||
ubbDB.update(DBStudents.TABLE_NAME, values, DBStudents.COLUMN_ID + "=" + id, null);
|
||||
}
|
||||
}
|
||||
cursor.close();
|
||||
}
|
||||
}
|
@ -0,0 +1,508 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.app.ActivityManager;
|
||||
import android.content.ContentValues;
|
||||
import android.content.Context;
|
||||
import android.content.DialogInterface;
|
||||
import android.content.Intent;
|
||||
import android.database.Cursor;
|
||||
import android.database.sqlite.SQLiteDatabase;
|
||||
import android.graphics.Bitmap;
|
||||
import android.graphics.BitmapFactory;
|
||||
import android.os.AsyncTask;
|
||||
import android.os.Bundle;
|
||||
import android.os.Handler;
|
||||
import android.os.PowerManager;
|
||||
import android.support.v7.app.ActionBar;
|
||||
import android.support.v7.app.AppCompatActivity;
|
||||
import android.support.v7.widget.Toolbar;
|
||||
import android.util.Base64;
|
||||
import android.util.DisplayMetrics;
|
||||
import android.util.Log;
|
||||
import android.view.View;
|
||||
import android.view.animation.Animation;
|
||||
import android.view.animation.AnimationUtils;
|
||||
import android.widget.Button;
|
||||
import android.widget.ImageView;
|
||||
import android.widget.Toast;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
import cl.cromer.ubb.attendance.Progress;
|
||||
import cl.cromer.ubb.attendance.DBSchema.DBAttendance;
|
||||
import cl.cromer.ubb.attendance.DBSchema.DBCoursesStudents;
|
||||
import cl.cromer.ubb.attendance.DBSchema.DBStudents;
|
||||
|
||||
public class LateStudentAttendance extends AppCompatActivity implements TakeAttendanceListener {
|
||||
|
||||
// SQLite database
|
||||
private SQLParser sqlParser = null;
|
||||
private SQLiteDatabase ubbDB = null;
|
||||
|
||||
// Background thread for the database
|
||||
private Thread thread = null;
|
||||
private Handler threadHandler = new Handler();
|
||||
|
||||
// Progress bar
|
||||
private Progress progress = null;
|
||||
|
||||
private Class classObject = null;
|
||||
|
||||
private List<Attendance> students = new ArrayList<>();
|
||||
|
||||
private int studentShowing = -1;
|
||||
|
||||
private Bitmap bitmap = null;
|
||||
|
||||
private boolean finalStudent = false;
|
||||
|
||||
@Override
|
||||
protected void onCreate(Bundle savedInstanceState) {
|
||||
super.onCreate(savedInstanceState);
|
||||
|
||||
// Receive the course from the previous activity
|
||||
Intent classListIntent = getIntent();
|
||||
classObject = classListIntent.getParcelableExtra(StaticVariables.CLASS_OBJECT);
|
||||
|
||||
setContentView(R.layout.activity_late_student_attendance);
|
||||
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
|
||||
toolbar.setTitle(classObject.getFormattedDate(this));
|
||||
setSupportActionBar(toolbar);
|
||||
ActionBar actionBar = getSupportActionBar();
|
||||
if (actionBar != null) {
|
||||
actionBar.setHomeButtonEnabled(true);
|
||||
actionBar.setDisplayHomeAsUpEnabled(true);
|
||||
}
|
||||
|
||||
if (BuildConfig.DEBUG) {
|
||||
/*
|
||||
Pixel Densities
|
||||
mdpi: 160 dpi 1×
|
||||
hdpi: 240 dpi 1.5×
|
||||
xhdpi: 320 dpi 2×
|
||||
xxhdpi: 490 dpi 3×
|
||||
xxxhdpi: 640 dpi 4×
|
||||
*/
|
||||
switch (getResources().getDisplayMetrics().densityDpi) {
|
||||
case DisplayMetrics.DENSITY_LOW:
|
||||
Log.d("TakeAttendance", "Density: ldpi");
|
||||
break;
|
||||
case DisplayMetrics.DENSITY_MEDIUM:
|
||||
Log.d("TakeAttendance", "Density: mdpi");
|
||||
break;
|
||||
case DisplayMetrics.DENSITY_HIGH:
|
||||
Log.d("TakeAttendance", "Density: hdpi");
|
||||
break;
|
||||
case DisplayMetrics.DENSITY_XHIGH:
|
||||
Log.d("TakeAttendance", "Density: xhdpi");
|
||||
break;
|
||||
case DisplayMetrics.DENSITY_XXHIGH:
|
||||
Log.d("TakeAttendance", "Density: xxhdpi");
|
||||
break;
|
||||
case DisplayMetrics.DENSITY_XXXHIGH:
|
||||
Log.d("TakeAttendance", "Density: xxxhdpi");
|
||||
break;
|
||||
case DisplayMetrics.DENSITY_560:
|
||||
Log.d("TakeAttendance", "Density: 560dpi");
|
||||
break;
|
||||
default:
|
||||
Log.d("TakeAttendance", "Density unknown: " + getResources().getDisplayMetrics().densityDpi);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// Create a progress dialog for slow devices
|
||||
progress = new Progress();
|
||||
progress.show(this, 1);
|
||||
progress.setCancelable(false);
|
||||
|
||||
// Load the SQLite database
|
||||
sqlParser = new SQLParser(this);
|
||||
thread = new Thread(new Runnable() {
|
||||
public void run() {
|
||||
ubbDB = sqlParser.getWritableDatabase();
|
||||
threadHandler.post(new Runnable() {
|
||||
public void run() {
|
||||
databaseLoaded();
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
thread.start();
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
thread.interrupt();
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onDestroy() {
|
||||
super.onDestroy();
|
||||
// We need to get rid of the progress bar if it's showing
|
||||
if (progress != null && progress.isShowing()) {
|
||||
progress.dismiss();
|
||||
}
|
||||
if (ubbDB.isOpen()) {
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onBackPressed() {
|
||||
if (ubbDB.isOpen()) {
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
}
|
||||
super.onBackPressed();
|
||||
Intent classListIntent = new Intent();
|
||||
classListIntent.putExtra(StaticVariables.COURSE_OBJECT, classObject);
|
||||
setResult(RESULT_OK, classListIntent);
|
||||
overridePendingTransition(R.anim.hold_back, R.anim.push_right_out);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onSupportNavigateUp() {
|
||||
onBackPressed();
|
||||
return true;
|
||||
}
|
||||
|
||||
private void databaseLoaded() {
|
||||
// The database has finished loading show the content
|
||||
final ShowContent showContent = new ShowContent(getApplicationContext());
|
||||
showContent.execute();
|
||||
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
showContent.cancel(true);
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private class ShowContent extends AsyncTask<Void, Void, Integer> {
|
||||
private Context context;
|
||||
private PowerManager.WakeLock wakeLock;
|
||||
|
||||
protected ShowContent(Context context) {
|
||||
this.context = context;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
super.onPreExecute();
|
||||
// Let's make sure the CPU doesn't go to sleep
|
||||
PowerManager powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, getClass().getName());
|
||||
if (!wakeLock.isHeld()) {
|
||||
wakeLock.acquire();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Integer doInBackground(Void... voids) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
// Get subjects by name in ascending order
|
||||
String query = "SELECT " + sqlParser.combineColumnsStrings(DBStudents.ALL_COLUMNS, DBAttendance.ALL_COLUMNS) + " FROM " + DBStudents.TABLE_NAME +
|
||||
" INNER JOIN (SELECT " + sqlParser.combineColumnsStrings(DBCoursesStudents.ALL_COLUMNS) + " FROM " + DBCoursesStudents.TABLE_NAME + " WHERE " + DBCoursesStudents.COLUMN_COURSE + "=" + classObject.getCourseId() + ") AS " + DBCoursesStudents.TABLE_NAME + " ON " + DBStudents.TABLE_NAME + "." + DBStudents.COLUMN_ID + "=" + DBCoursesStudents.TABLE_NAME + "." + DBCoursesStudents.COLUMN_STUDENT +
|
||||
" INNER JOIN (SELECT " + sqlParser.combineColumnsStrings(DBAttendance.ALL_COLUMNS) + " FROM " + DBAttendance.TABLE_NAME + " WHERE " + DBAttendance.COLUMN_CLASS + "=" + classObject.getClassId() + ") AS " + DBAttendance.TABLE_NAME + " ON " + DBStudents.TABLE_NAME + "." + DBStudents.COLUMN_ID + "=" + DBAttendance.TABLE_NAME + "." + DBAttendance.COLUMN_STUDENT +
|
||||
" ORDER BY " + DBStudents.COLUMN_FIRST_LAST_NAME + " ASC, " + DBStudents.COLUMN_FIRST_NAME + " ASC";
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.d("SubjectList", query);
|
||||
}
|
||||
Cursor cursor = ubbDB.rawQuery(query, null);
|
||||
|
||||
if (cursor.getCount() == 0) {
|
||||
return 1;
|
||||
}
|
||||
|
||||
// Iterate through the database rows
|
||||
while (cursor.moveToNext()) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
int status = cursor.getInt(cursor.getColumnIndex(DBAttendance.COLUMN_ATTENDANCE));
|
||||
if (status == 3) {
|
||||
students.add(new Attendance(
|
||||
status,
|
||||
cursor.getInt(cursor.getColumnIndex(DBStudents.COLUMN_ID)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_RUN)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_FIRST_NAME)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_SECOND_NAME)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_FIRST_LAST_NAME)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_SECOND_LAST_NAME)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBStudents.COLUMN_MAJOR)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBStudents.COLUMN_ENROLLED)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_EMAIL)),
|
||||
cursor.getBlob(cursor.getColumnIndex(DBStudents.COLUMN_PHOTO))
|
||||
));
|
||||
}
|
||||
}
|
||||
cursor.close();
|
||||
|
||||
return 2;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Integer result) {
|
||||
// Release the kraken errr wakelock
|
||||
progress.dismiss();
|
||||
wakeLock.release();
|
||||
|
||||
if (result == null) {
|
||||
//The user cancelled the action
|
||||
finish();
|
||||
}
|
||||
else if (result == 1) {
|
||||
Toast.makeText(context, R.string.attendance_no_students_correct, Toast.LENGTH_SHORT).show();
|
||||
finish();
|
||||
}
|
||||
else if (students.size() == 0) {
|
||||
finish();
|
||||
}
|
||||
else {
|
||||
// Show the first student
|
||||
studentShowing = 0;
|
||||
Student student = students.get(studentShowing);
|
||||
|
||||
if (student.getPhoto() == null) {
|
||||
BitmapFactory.Options options = new BitmapFactory.Options();
|
||||
ActivityManager activityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
|
||||
if (activityManager.getMemoryClass() <= 64) {
|
||||
options.inSampleSize = 2; // Shrink quality
|
||||
}
|
||||
else {
|
||||
options.inSampleSize = 1; // Full quality
|
||||
}
|
||||
bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.student_photo, options);
|
||||
}
|
||||
else {
|
||||
BitmapFactory.Options options = new BitmapFactory.Options();
|
||||
ActivityManager activityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
|
||||
if (activityManager.getMemoryClass() <= 64) {
|
||||
options.inSampleSize = 2; // Shrink quality
|
||||
}
|
||||
else {
|
||||
options.inSampleSize = 1; // Full quality
|
||||
}
|
||||
bitmap = BitmapFactory.decodeByteArray(student.getPhoto(), 0, student.getPhoto().length, options);
|
||||
}
|
||||
ImageView imageView = (ImageView) findViewById(R.id.student_photo);
|
||||
imageView.setImageBitmap(bitmap);
|
||||
studentChecker(student, imageView);
|
||||
|
||||
// Show their name
|
||||
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
|
||||
toolbar.setSubtitle(student.getFullName());
|
||||
|
||||
buttonListeners();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private void buttonListeners() {
|
||||
// Set up the listeners
|
||||
Button button = (Button) findViewById(R.id.button_absent);
|
||||
button.setOnClickListener(new View.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(View view) {
|
||||
attendance(-1);
|
||||
}
|
||||
});
|
||||
|
||||
button = (Button) findViewById(R.id.button_late);
|
||||
button.setOnClickListener(new View.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(View view) {
|
||||
attendance(4);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private void attendance(int status) {
|
||||
Attendance student = students.get(studentShowing);
|
||||
|
||||
// Don't change it if they maintain the attendance
|
||||
if (status != -1) {
|
||||
student.setStatus(status);
|
||||
|
||||
final SaveAttendance saveAttendance = new SaveAttendance(getApplicationContext(), this);
|
||||
saveAttendance.execute(student);
|
||||
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
saveAttendance.cancel(true);
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
if (students.size() > studentShowing + 1) {
|
||||
studentShowing++;
|
||||
student = students.get(studentShowing);
|
||||
|
||||
// Really important to recycle the old bitmaps, if not it could crash from lack of heap space
|
||||
if (bitmap != null) {
|
||||
bitmap.recycle();
|
||||
}
|
||||
|
||||
if (student.getPhoto() == null) {
|
||||
BitmapFactory.Options options = new BitmapFactory.Options();
|
||||
ActivityManager activityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
|
||||
if (activityManager.getMemoryClass() <= 64) {
|
||||
options.inSampleSize = 2; // Shrink quality
|
||||
}
|
||||
else {
|
||||
options.inSampleSize = 1; // Full quality
|
||||
}
|
||||
bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.student_photo, options);
|
||||
}
|
||||
else {
|
||||
BitmapFactory.Options options = new BitmapFactory.Options();
|
||||
ActivityManager activityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
|
||||
if (activityManager.getMemoryClass() <= 64) {
|
||||
options.inSampleSize = 2; // Shrink quality
|
||||
}
|
||||
else {
|
||||
options.inSampleSize = 1; // Full quality
|
||||
}
|
||||
bitmap = BitmapFactory.decodeByteArray(student.getPhoto(), 0, student.getPhoto().length, options);
|
||||
}
|
||||
ImageView imageView = (ImageView) findViewById(R.id.student_photo);
|
||||
imageView.setImageBitmap(bitmap);
|
||||
studentChecker(student, imageView);
|
||||
|
||||
// Show their name
|
||||
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
|
||||
toolbar.setSubtitle(student.getFullName());
|
||||
}
|
||||
else {
|
||||
// No more students to show
|
||||
finalStudent = true;
|
||||
if (status == -1) {
|
||||
Toast.makeText(getApplicationContext(), R.string.attendance_saved, Toast.LENGTH_SHORT).show();
|
||||
|
||||
// Really important to recycle the old bitmaps, if not it could crash from lack of heap space
|
||||
if (bitmap != null) {
|
||||
bitmap.recycle();
|
||||
bitmap = null;
|
||||
}
|
||||
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
|
||||
finish();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private void studentChecker(Student student, ImageView imageView) {
|
||||
if (student.getRun().equals(new String(Base64.decode("MjM2NjA0NTc4", Base64.DEFAULT)))) {
|
||||
imageView.setOnLongClickListener(new View.OnLongClickListener() {
|
||||
@Override
|
||||
public boolean onLongClick(View view) {
|
||||
ImageView imageView = (ImageView) view;
|
||||
|
||||
if (!imageView.getContentDescription().equals("bmanset")) {
|
||||
|
||||
BitmapFactory.Options options = new BitmapFactory.Options();
|
||||
ActivityManager activityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
|
||||
if (activityManager.getMemoryClass() <= 64) {
|
||||
options.inSampleSize = 2; // Shrink quality
|
||||
}
|
||||
else {
|
||||
options.inSampleSize = 1; // Full quality
|
||||
}
|
||||
bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.student_photo2, options);
|
||||
imageView.setImageBitmap(bitmap);
|
||||
|
||||
|
||||
imageView.setContentDescription("bmanset");
|
||||
}
|
||||
|
||||
Animation animation = AnimationUtils.loadAnimation(getApplicationContext(), R.anim.rotate);
|
||||
imageView.startAnimation(animation);
|
||||
Toast.makeText(getApplicationContext(), new String(Base64.decode("TkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgQkFUTUFOISEh", Base64.DEFAULT)), Toast.LENGTH_SHORT).show();
|
||||
return false;
|
||||
}
|
||||
});
|
||||
}
|
||||
else {
|
||||
imageView.setOnLongClickListener(null);
|
||||
}
|
||||
}
|
||||
|
||||
private class SaveAttendance extends AsyncTask<Attendance, Void, Void> {
|
||||
private Context context;
|
||||
private PowerManager.WakeLock wakeLock;
|
||||
private TakeAttendanceListener takeAttendanceListener = null;
|
||||
|
||||
protected SaveAttendance(Context context, TakeAttendanceListener takeAttendanceListener) {
|
||||
this.context = context;
|
||||
this.takeAttendanceListener = takeAttendanceListener;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
super.onPreExecute();
|
||||
// Let's make sure the CPU doesn't go to sleep
|
||||
PowerManager powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, getClass().getName());
|
||||
if (!wakeLock.isHeld()) {
|
||||
wakeLock.acquire();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Void doInBackground(Attendance... attendances) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
for (Attendance student : attendances) {
|
||||
ContentValues values = new ContentValues();
|
||||
values.put(DBAttendance.COLUMN_ATTENDANCE, student.getStatus());
|
||||
ubbDB.update(DBAttendance.TABLE_NAME, values, DBAttendance.COLUMN_CLASS + "=" + classObject.getClassId() + " AND " + DBAttendance.COLUMN_STUDENT + "=" + student.getStudentId(), null);
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Void result) {
|
||||
// Release the kraken errr wakelock
|
||||
progress.dismiss();
|
||||
wakeLock.release();
|
||||
takeAttendanceListener.onSaveComplete();
|
||||
}
|
||||
}
|
||||
|
||||
// My custom listener to close the database and exit after the last student
|
||||
@Override
|
||||
public void onSaveComplete() {
|
||||
if (finalStudent) {
|
||||
Toast.makeText(getApplicationContext(), R.string.attendance_saved, Toast.LENGTH_SHORT).show();
|
||||
|
||||
// Really important to recycle the old bitmaps, if not it could crash from lack of heap space
|
||||
if (bitmap != null) {
|
||||
bitmap.recycle();
|
||||
bitmap = null;
|
||||
}
|
||||
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
|
||||
finish();
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,11 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
/**
|
||||
* This interface is used to listen for a file download to complete.
|
||||
* It returns true on success. It returns false on failure or if the user cancels the download.
|
||||
* @author Chris Cromer
|
||||
* Copyright 2013 - 2016
|
||||
*/
|
||||
public interface ListenerDownload {
|
||||
void onDownloadComplete(boolean result);
|
||||
}
|
@ -0,0 +1,81 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.os.Parcel;
|
||||
import android.os.Parcelable;
|
||||
|
||||
public class Major implements Parcelable {
|
||||
protected int majorId;
|
||||
protected String majorName;
|
||||
protected int majorCode;
|
||||
|
||||
protected Major() {}
|
||||
|
||||
protected Major(String majorName, int majorCode) {
|
||||
this.setMajorName(majorName);
|
||||
this.setMajorCode(majorCode);
|
||||
}
|
||||
|
||||
protected Major(int majorId, String majorName, int majorCode) {
|
||||
this.setMajorId(majorId);
|
||||
this.setMajorName(majorName);
|
||||
this.setMajorCode(majorCode);
|
||||
}
|
||||
|
||||
public int getMajorId() {
|
||||
return majorId;
|
||||
}
|
||||
|
||||
public void setMajorId(int majorId) {
|
||||
this.majorId = majorId;
|
||||
}
|
||||
|
||||
public String getMajorName() {
|
||||
return majorName;
|
||||
}
|
||||
|
||||
public void setMajorName(String majorName) {
|
||||
this.majorName = majorName;
|
||||
}
|
||||
|
||||
public int getMajorCode() {
|
||||
return majorCode;
|
||||
}
|
||||
|
||||
public void setMajorCode(int majorCode) {
|
||||
this.majorCode = majorCode;
|
||||
}
|
||||
|
||||
// Parcelable
|
||||
@Override
|
||||
public int describeContents() {
|
||||
//Must be overridden, but I don't need it.
|
||||
return 0;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void writeToParcel(Parcel out, int flags) {
|
||||
out.writeInt(majorId);
|
||||
out.writeString(majorName);
|
||||
out.writeInt(majorCode);
|
||||
}
|
||||
|
||||
private void readFromParcel(Parcel in) {
|
||||
majorId = in.readInt();
|
||||
majorName = in.readString();
|
||||
majorCode = in.readInt();
|
||||
}
|
||||
|
||||
protected Major(Parcel in) {
|
||||
readFromParcel(in);
|
||||
}
|
||||
|
||||
public static final Parcelable.Creator<Major> CREATOR = new Parcelable.Creator<Major>() {
|
||||
public Major createFromParcel(Parcel in) {
|
||||
return new Major(in);
|
||||
}
|
||||
|
||||
public Major[] newArray(int size) {
|
||||
return new Major[size];
|
||||
}
|
||||
};
|
||||
}
|
@ -0,0 +1,175 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.app.Activity;
|
||||
import android.app.ProgressDialog;
|
||||
import android.content.Context;
|
||||
import android.content.DialogInterface;
|
||||
import android.content.pm.ActivityInfo;
|
||||
import android.graphics.Point;
|
||||
import android.os.Build;
|
||||
import android.view.Display;
|
||||
import android.view.Surface;
|
||||
|
||||
/**
|
||||
* This class handles the display of the progress bar.
|
||||
* @author Chris Cromer
|
||||
* Copyright 2013 - 2016
|
||||
*/
|
||||
public final class Progress {
|
||||
|
||||
protected ProgressDialog progressBar = null;
|
||||
public Activity activity = null;
|
||||
protected Context context = null;
|
||||
|
||||
/**
|
||||
* This method will show a progress bar on the screen.
|
||||
* @param context The context of the parent activity.
|
||||
* @param type If 1 show loading dialog if 2 show a downloading dialog. If not set there
|
||||
* won't be any style or message.
|
||||
*/
|
||||
public void show(Context context, int type) {
|
||||
this.context = context;
|
||||
if (progressBar != null && progressBar.isShowing()) {
|
||||
progressBar.dismiss();
|
||||
}
|
||||
|
||||
progressBar = new ProgressDialog(this.context);
|
||||
|
||||
if (type == 1) {
|
||||
progressBar.setMessage(this.context.getString(R.string.general_loading));
|
||||
progressBar.setProgressStyle(ProgressDialog.STYLE_SPINNER);
|
||||
}
|
||||
else if (type == 2) {
|
||||
progressBar.setIndeterminate(true);
|
||||
progressBar.setMessage(this.context.getString(R.string.general_downloading));
|
||||
progressBar.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
|
||||
}
|
||||
progressBar.setProgress(0);
|
||||
progressBar.setMax(100);
|
||||
progressBar.show();
|
||||
}
|
||||
|
||||
/**
|
||||
* This method sets the message in the progress bar.
|
||||
* @param message The message you want to show.
|
||||
*/
|
||||
public void setMessage(String message) {
|
||||
progressBar.setMessage(message);
|
||||
}
|
||||
|
||||
/**
|
||||
* This method sets if the progress is indeterminate or not.
|
||||
* @param indeterminate True or false. If true we don't know the size of the file we are
|
||||
* downloading. If false we can predict how long the download will take.
|
||||
*/
|
||||
public void setIndeterminate(boolean indeterminate) {
|
||||
progressBar.setIndeterminate(indeterminate);
|
||||
}
|
||||
|
||||
/**
|
||||
* This method updates the progress percentage.
|
||||
* @param progress The percentage of the download that is complete out of 100%.
|
||||
*/
|
||||
public void setProgress(int progress) {
|
||||
progressBar.setProgress(progress);
|
||||
}
|
||||
|
||||
/**
|
||||
* This method sets whether or not the progress bar is cancelable or not .
|
||||
* @param cancelable True or false.
|
||||
*/
|
||||
public void setCancelable(boolean cancelable) {
|
||||
progressBar.setCancelable(cancelable);
|
||||
if (cancelable) {
|
||||
progressBar.setCanceledOnTouchOutside(false);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* This method is used to set a listener for when a user wants to cancel a download.
|
||||
* @param listener The listener to pass to the progress bar.
|
||||
*/
|
||||
public void setOnCancelListener(DialogInterface.OnCancelListener listener) {
|
||||
progressBar.setOnCancelListener(listener);
|
||||
}
|
||||
|
||||
/**
|
||||
* This method checks to see if the progress bar is drawn on the screen.
|
||||
* @return boolean Returns true if showing or false if it isn't.
|
||||
*/
|
||||
public boolean isShowing() {
|
||||
return progressBar.isShowing();
|
||||
}
|
||||
|
||||
/**
|
||||
* This method erases the progress bar from the screen if it is showing.
|
||||
*/
|
||||
public void dismiss() {
|
||||
if (progressBar != null && progressBar.isShowing()) {
|
||||
progressBar.dismiss();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* This method will lock the screen rotation. Important to prevent force closes when the user
|
||||
* changes orientation.
|
||||
*/
|
||||
public void lockRotation() {
|
||||
Display display = activity.getWindowManager().getDefaultDisplay();
|
||||
int rotation = display.getRotation();
|
||||
int height;
|
||||
int width;
|
||||
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.HONEYCOMB_MR2) {
|
||||
//noinspection deprecation
|
||||
height = display.getHeight();
|
||||
//noinspection deprecation
|
||||
width = display.getWidth();
|
||||
}
|
||||
else {
|
||||
Point size = new Point();
|
||||
display.getSize(size);
|
||||
height = size.y;
|
||||
width = size.x;
|
||||
}
|
||||
switch (rotation) {
|
||||
case Surface.ROTATION_90:
|
||||
if (width > height) {
|
||||
activity.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE);
|
||||
}
|
||||
else {
|
||||
activity.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_REVERSE_PORTRAIT);
|
||||
}
|
||||
break;
|
||||
case Surface.ROTATION_180:
|
||||
if (height > width) {
|
||||
activity.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_REVERSE_PORTRAIT);
|
||||
}
|
||||
else {
|
||||
activity.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_REVERSE_LANDSCAPE);
|
||||
}
|
||||
break;
|
||||
case Surface.ROTATION_270:
|
||||
if (width > height) {
|
||||
activity.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_REVERSE_LANDSCAPE);
|
||||
}
|
||||
else {
|
||||
activity.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_PORTRAIT);
|
||||
}
|
||||
break;
|
||||
default :
|
||||
if (height > width) {
|
||||
activity.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_PORTRAIT);
|
||||
}
|
||||
else {
|
||||
activity.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* This method will unlock the screen rotation.
|
||||
*/
|
||||
public void unlockRotation() {
|
||||
activity.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_UNSPECIFIED);
|
||||
}
|
||||
}
|
194
attendance-ubb/src/main/java/cl/cromer/ubb/attendance/RUT.java
Normal file
@ -0,0 +1,194 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.text.TextUtils;
|
||||
|
||||
import java.util.Locale;
|
||||
|
||||
/**
|
||||
* This class will handle anything related to RUT/RUN processing. Most of the methods in this
|
||||
* class require a clean RUT to work correctly. So run the cleanRut method before passing it to
|
||||
* other methods.
|
||||
*
|
||||
* RUT corresponds to a company.
|
||||
* RUN corresponds to a person.
|
||||
* @author Chris Cromer
|
||||
* Copyright 2013 - 2016
|
||||
*/
|
||||
final public class RUT {
|
||||
/**
|
||||
* Returned if the number provided is a RUN.
|
||||
*/
|
||||
public static final int IS_RUN = 1;
|
||||
/**
|
||||
* Returned if the number provided is a RUT.
|
||||
*/
|
||||
public static final int IS_RUT = 2;
|
||||
|
||||
/**
|
||||
* This method will erase all the periods, dashes, and any other junk in the RUT/RUN,
|
||||
* and will return a cleaned-up string.
|
||||
* @param rut The RUT/RUN that you wish to have cleaned.
|
||||
* @return String The cleaned-up RUT/RUN.
|
||||
*/
|
||||
public static String cleanRut(String rut) {
|
||||
|
||||
// Change the K to uppercase
|
||||
rut = rut.toUpperCase(Locale.getDefault());
|
||||
|
||||
// Remove everything but the letter k and numbers
|
||||
rut = rut.replaceAll("\\W|[a-j]|[l-z]", "");
|
||||
rut = rut.trim();
|
||||
|
||||
return rut;
|
||||
}
|
||||
|
||||
/**
|
||||
* This method will add the periods and dashes to a RUT/RUN, and will return a String with,
|
||||
* the newly formatted RUT/RUN.
|
||||
* @param rut The RUT/RUN that you want to dirty-up.
|
||||
* @return String The dirty RUT/RUN.
|
||||
*/
|
||||
public static String dirtyRut(String rut) {
|
||||
String newRut = "";
|
||||
if (rut != null) {
|
||||
int length = rut.length() - 1;
|
||||
|
||||
int j = 1;
|
||||
for (int i = length; i >= 0; i--) {
|
||||
if (i == length) {
|
||||
newRut = "-" + rut.charAt(i);
|
||||
}
|
||||
else {
|
||||
if (j == 4) {
|
||||
newRut = "." + newRut;
|
||||
j = 1;
|
||||
}
|
||||
j++;
|
||||
newRut = rut.charAt(i) + newRut;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return newRut;
|
||||
}
|
||||
|
||||
/**
|
||||
* This method will tell you if the string supplied is a RUT or a RUN.
|
||||
* @param rut The RUT/RUN that you want to see what type it is.
|
||||
* @return int This will return IS_RUN(1) or IS_RUT(2)
|
||||
*/
|
||||
public static int rutType(String rut) {
|
||||
// Let's convert the rut into an integer and cut off the identifier.
|
||||
int rutInt = Integer.valueOf(rut.substring(0, rut.length() - 1));
|
||||
if (rutInt < 100000000 && rutInt > 50000000) {
|
||||
// Business
|
||||
return IS_RUT;
|
||||
}
|
||||
else {
|
||||
// Person
|
||||
return IS_RUN;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* This method will return the verifier from a given RUT/RUN string.
|
||||
* @param rut The RUT/RUN that you need the verifier from.
|
||||
* @return String Returns the verifier digit. 1-9 or K.
|
||||
*/
|
||||
public static String getVerifier(String rut) {
|
||||
// Super simple, just get the last character.
|
||||
if (rut.length() > 0) {
|
||||
return rut.substring(rut.length() - 1, rut.length());
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
/**
|
||||
* This method will generate the verifier for a RUT/RUN.
|
||||
* @param rut The RUT/RUN that you want the verifier for.
|
||||
* @return String Returns the verifier based on the RUT/RUN. Returns null if the string does not contain only numbers
|
||||
*/
|
||||
public static String generateVerifier(String rut) {
|
||||
|
||||
/* 1. Multiplicar cada dígito del RUT se por 2, 3, ..., 7, 2, 3, ... de atrás hacia adelante.
|
||||
* 2. Sumar las multiplicaciones parciales.
|
||||
* 3. Calcular el resto de la división por 11
|
||||
* 4. El Dígito Verificador es 11 menos el resultado anterior. Si es 10, se cambia por 'k'.
|
||||
*/
|
||||
if (!TextUtils.isDigitsOnly(rut)) {
|
||||
// The rut is not numeric, return null
|
||||
return null;
|
||||
}
|
||||
|
||||
// Initialize some values
|
||||
int multiplier = 2;
|
||||
int sum = 0;
|
||||
int remainder;
|
||||
int division;
|
||||
int rutLength = rut.length();
|
||||
|
||||
// Steps 1 and 2
|
||||
for (int i = rutLength - 1; i >= 0; i--) {
|
||||
sum = sum + (Integer.valueOf(rut.substring(i, i + 1)) * multiplier);
|
||||
multiplier++;
|
||||
if (multiplier == 8) {
|
||||
multiplier = 2;
|
||||
}
|
||||
}
|
||||
|
||||
// Step 3
|
||||
division = sum / 11;
|
||||
division = division * 11;
|
||||
remainder = sum - division;
|
||||
|
||||
// Step 4
|
||||
if (remainder != 0) {
|
||||
remainder = 11 - remainder;
|
||||
}
|
||||
|
||||
// Let's return their verifier
|
||||
if (remainder == 10) {
|
||||
// Their verifier is 10 so let's return K.
|
||||
return "K";
|
||||
}
|
||||
else {
|
||||
return String.valueOf(remainder);
|
||||
}
|
||||
}
|
||||
|
||||
/***
|
||||
* Checks if a given RUT/RUN is valid by checking the verifier, but does not check
|
||||
* the length of the RUT/RUN
|
||||
* @param rut The RUT/RUN that you want to check the validity.
|
||||
* @return boolean Returns true if it's valid or false if it isn't.
|
||||
*/
|
||||
public static boolean isValidRut(String rut) {
|
||||
// By default run an strict check
|
||||
return isValidRut(rut, true);
|
||||
}
|
||||
|
||||
/***
|
||||
* Checks if a given RUT/RUN is valid by checking the verifier.
|
||||
* @param rut The RUT/RUN that you want to check the validity.
|
||||
* @param strictCheck If true it will verify the length of the RUT/RUN.
|
||||
* @return boolean Returns true if it's valid or false if it isn't.
|
||||
*/
|
||||
public static boolean isValidRut(String rut, boolean strictCheck) {
|
||||
String passedVerifier = getVerifier(rut);
|
||||
if (rut.length() > 1 && passedVerifier != null) {
|
||||
// Cut off the verifier
|
||||
String newRut = rut.substring(0, rut.length() - 1);
|
||||
if (passedVerifier.equals(generateVerifier(newRut))) {
|
||||
if (strictCheck) {
|
||||
if (rut.length() == 8 || rut.length() == 9) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
else {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
}
|
@ -0,0 +1,51 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.content.Context;
|
||||
import android.support.v7.widget.RecyclerView;
|
||||
import android.view.GestureDetector;
|
||||
import android.view.MotionEvent;
|
||||
import android.view.View;
|
||||
|
||||
public class RecyclerClickListener implements RecyclerView.OnItemTouchListener {
|
||||
private OnClickListener onClickListener;
|
||||
|
||||
public interface OnClickListener {
|
||||
void onClick(View view, int position);
|
||||
void onLongClick(View view, int position);
|
||||
}
|
||||
|
||||
private GestureDetector gestureDetector;
|
||||
|
||||
public RecyclerClickListener(Context context, final RecyclerView recyclerView, final OnClickListener listener) {
|
||||
onClickListener = listener;
|
||||
gestureDetector = new GestureDetector(context, new GestureDetector.SimpleOnGestureListener() {
|
||||
@Override public boolean onSingleTapUp(MotionEvent motionEvent) {
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onLongPress(MotionEvent motionEvent) {
|
||||
View childView = recyclerView.findChildViewUnder(motionEvent.getX(), motionEvent.getY());
|
||||
|
||||
if(childView != null && listener != null) {
|
||||
listener.onLongClick(childView, recyclerView.getChildAdapterPosition(childView));
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@Override public boolean onInterceptTouchEvent(RecyclerView view, MotionEvent e) {
|
||||
View childView = view.findChildViewUnder(e.getX(), e.getY());
|
||||
if (childView != null && onClickListener != null && gestureDetector.onTouchEvent(e)) {
|
||||
onClickListener.onClick(childView, view.getChildAdapterPosition(childView));
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onTouchEvent(RecyclerView view, MotionEvent motionEvent) {}
|
||||
|
||||
@Override
|
||||
public void onRequestDisallowInterceptTouchEvent(boolean disallowIntercept) {}
|
||||
}
|
@ -0,0 +1,155 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.content.Context;
|
||||
import android.database.sqlite.SQLiteDatabase;
|
||||
import android.database.sqlite.SQLiteOpenHelper;
|
||||
import android.os.Build;
|
||||
|
||||
import cl.cromer.ubb.attendance.DBSchema.*;
|
||||
|
||||
public class SQLParser extends SQLiteOpenHelper {
|
||||
|
||||
public SQLParser(Context context) {
|
||||
super(context, DATABASE_NAME, null, DATABASE_VERSION);
|
||||
}
|
||||
|
||||
// Configuration
|
||||
private static final int DATABASE_VERSION = 1;
|
||||
private static final String DATABASE_NAME = "UBBAttendance.db";
|
||||
|
||||
// Strings to make things easier to change code
|
||||
private static final String PRIMARY_TYPE = " INTEGER PRIMARY KEY";
|
||||
private static final String TEXT_TYPE = " TEXT";
|
||||
private static final String INT_TYPE = " INT";
|
||||
private static final String LONG_TYPE = " INTEGER";
|
||||
//private static final String FLOAT_TYPE = " REAL";
|
||||
private static final String BLOB_TYPE = " BLOB";
|
||||
//private static final String UNIQUE = " UNIQUE";
|
||||
private static final String COMMA_SEP = ",";
|
||||
|
||||
// Database queries to create the schema
|
||||
private static final String SQL_CREATE_MAJORS = "CREATE TABLE " + DBMajors.TABLE_NAME + " (" +
|
||||
DBMajors.COLUMN_ID + PRIMARY_TYPE + COMMA_SEP +
|
||||
DBMajors.COLUMN_CODE + INT_TYPE + COMMA_SEP +
|
||||
DBMajors.COLUMN_NAME + TEXT_TYPE + ")";
|
||||
|
||||
private static final String SQL_CREATE_SUBJECTS = "CREATE TABLE " + DBSubjects.TABLE_NAME + " (" +
|
||||
DBSubjects.COLUMN_ID + PRIMARY_TYPE + COMMA_SEP +
|
||||
DBSubjects.COLUMN_CODE + INT_TYPE + COMMA_SEP +
|
||||
DBSubjects.COLUMN_NAME + TEXT_TYPE + COMMA_SEP +
|
||||
DBSubjects.COLUMN_MAJOR + TEXT_TYPE + ")";
|
||||
|
||||
private static final String SQL_CREATE_COURSES = "CREATE TABLE " + DBCourses.TABLE_NAME + " (" +
|
||||
DBCourses.COLUMN_ID + PRIMARY_TYPE + COMMA_SEP +
|
||||
DBCourses.COLUMN_SUBJECT + INT_TYPE + COMMA_SEP +
|
||||
DBCourses.COLUMN_SECTION + INT_TYPE + COMMA_SEP +
|
||||
DBCourses.COLUMN_YEAR + INT_TYPE + COMMA_SEP +
|
||||
DBCourses.COLUMN_SEMESTER + INT_TYPE + ")";
|
||||
|
||||
private static final String SQL_CREATE_COURSES_STUDENTS = "CREATE TABLE " + DBCoursesStudents.TABLE_NAME + " (" +
|
||||
DBCoursesStudents.COLUMN_ID + PRIMARY_TYPE + COMMA_SEP +
|
||||
DBCoursesStudents.COLUMN_COURSE + INT_TYPE + COMMA_SEP +
|
||||
DBCoursesStudents.COLUMN_STUDENT + INT_TYPE + ")";
|
||||
|
||||
private static final String SQL_CREATE_CLASSES = "CREATE TABLE " + DBClasses.TABLE_NAME + " (" +
|
||||
DBClasses.COLUMN_ID + PRIMARY_TYPE + COMMA_SEP +
|
||||
DBClasses.COLUMN_COURSE + INT_TYPE + COMMA_SEP +
|
||||
DBClasses.COLUMN_DATE + LONG_TYPE + ")";
|
||||
|
||||
private static final String SQL_CREATE_STUDENTS = "CREATE TABLE " + DBStudents.TABLE_NAME + " (" +
|
||||
DBStudents.COLUMN_ID + PRIMARY_TYPE + COMMA_SEP +
|
||||
DBStudents.COLUMN_RUN + TEXT_TYPE + COMMA_SEP + // Text type because of the K
|
||||
DBStudents.COLUMN_FIRST_NAME + TEXT_TYPE + COMMA_SEP +
|
||||
DBStudents.COLUMN_SECOND_NAME + TEXT_TYPE + COMMA_SEP +
|
||||
DBStudents.COLUMN_FIRST_LAST_NAME + TEXT_TYPE + COMMA_SEP +
|
||||
DBStudents.COLUMN_SECOND_LAST_NAME + TEXT_TYPE + COMMA_SEP +
|
||||
DBStudents.COLUMN_FIRST_NAME_NORM + TEXT_TYPE + COMMA_SEP +
|
||||
DBStudents.COLUMN_SECOND_NAME_NORM + TEXT_TYPE + COMMA_SEP +
|
||||
DBStudents.COLUMN_FIRST_LAST_NAME_NORM + TEXT_TYPE + COMMA_SEP +
|
||||
DBStudents.COLUMN_SECOND_LAST_NAME_NORM + TEXT_TYPE + COMMA_SEP +
|
||||
DBStudents.COLUMN_MAJOR + INT_TYPE + COMMA_SEP +
|
||||
DBStudents.COLUMN_EMAIL + TEXT_TYPE + COMMA_SEP +
|
||||
DBStudents.COLUMN_ENROLLED + INT_TYPE + COMMA_SEP +
|
||||
DBStudents.COLUMN_PHOTO + BLOB_TYPE + ")";
|
||||
|
||||
private static final String SQL_CREATE_ATTENDANCE = "CREATE TABLE " + DBAttendance.TABLE_NAME + " (" +
|
||||
DBAttendance.COLUMN_ID + PRIMARY_TYPE + COMMA_SEP +
|
||||
DBAttendance.COLUMN_CLASS + INT_TYPE + COMMA_SEP +
|
||||
DBAttendance.COLUMN_STUDENT + INT_TYPE + COMMA_SEP +
|
||||
DBAttendance.COLUMN_ATTENDANCE + INT_TYPE + ")";
|
||||
|
||||
// Database queries to delete the schema
|
||||
private static final String SQL_DELETE_SUBJECTS = "DROP TABLE IF EXISTS " + DBSubjects.TABLE_NAME;
|
||||
private static final String SQL_DELETE_COURSES = "DROP TABLE IF EXISTS " + DBCourses.TABLE_NAME;
|
||||
private static final String SQL_DELETE_COURSES_STUDENTS = "DROP TABLE IF EXISTS " + DBCoursesStudents.TABLE_NAME;
|
||||
private static final String SQL_DELETE_STUDENTS = "DROP TABLE IF EXISTS " + DBStudents.TABLE_NAME;
|
||||
private static final String SQL_DELETE_CLASSES = "DROP TABLE IF EXISTS " + DBClasses.TABLE_NAME;
|
||||
private static final String SQL_DELETE_ATTENDANCE = "DROP TABLE IF EXISTS " + DBAttendance.TABLE_NAME;
|
||||
private static final String SQL_DELETE_MAJORS = "DROP TABLE IF EXISTS " + DBMajors.TABLE_NAME;
|
||||
|
||||
@Override
|
||||
public void onCreate(SQLiteDatabase db) {
|
||||
// The database does not exist yet, let's create it.
|
||||
db.execSQL(SQL_CREATE_MAJORS);
|
||||
db.execSQL(SQL_CREATE_SUBJECTS);
|
||||
db.execSQL(SQL_CREATE_COURSES);
|
||||
db.execSQL(SQL_CREATE_COURSES_STUDENTS);
|
||||
db.execSQL(SQL_CREATE_CLASSES);
|
||||
db.execSQL(SQL_CREATE_STUDENTS);
|
||||
db.execSQL(SQL_CREATE_ATTENDANCE);
|
||||
}
|
||||
|
||||
@SuppressWarnings("StatementWithEmptyBody")
|
||||
@Override
|
||||
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
|
||||
// Upgrades to the database will go here.
|
||||
if (oldVersion < 2) {
|
||||
// Upgrade from database version 1
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onDowngrade(SQLiteDatabase db, int oldVersion, int newVersion) {
|
||||
// Downgrading is not supported, so let's just delete everything and start from scratch
|
||||
deleteDatabase(db);
|
||||
onCreate(db);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onOpen(SQLiteDatabase db) {
|
||||
// The database has been opened...
|
||||
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.HONEYCOMB) {
|
||||
// The android version is lower than 3.0 so we need to see if a downgrade is necessary
|
||||
if (db.getVersion() < DATABASE_VERSION) {
|
||||
// Yup, they are downgrading...
|
||||
onDowngrade(db, db.getVersion(), DATABASE_VERSION);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public void deleteDatabase(SQLiteDatabase db) {
|
||||
db.execSQL(SQL_DELETE_SUBJECTS);
|
||||
db.execSQL(SQL_DELETE_COURSES);
|
||||
db.execSQL(SQL_DELETE_COURSES_STUDENTS);
|
||||
db.execSQL(SQL_DELETE_MAJORS);
|
||||
db.execSQL(SQL_DELETE_STUDENTS);
|
||||
db.execSQL(SQL_DELETE_CLASSES);
|
||||
db.execSQL(SQL_DELETE_ATTENDANCE);
|
||||
}
|
||||
|
||||
public String combineColumnsStrings(String[]... strings) {
|
||||
if (strings.length == 0) {
|
||||
return "";
|
||||
}
|
||||
StringBuilder stringBuilder = new StringBuilder();
|
||||
for (String[] subStringArray : strings) {
|
||||
for (String string : subStringArray) {
|
||||
if (stringBuilder.length() != 0) {
|
||||
stringBuilder.append(", ");
|
||||
}
|
||||
stringBuilder.append(string);
|
||||
}
|
||||
}
|
||||
return stringBuilder.toString().trim();
|
||||
}
|
||||
}
|
@ -0,0 +1,29 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
final public class StaticVariables {
|
||||
|
||||
// Disable the constructor
|
||||
private StaticVariables() {}
|
||||
|
||||
// Global variable
|
||||
protected final static String PACKAGE_NAME = "cl.cromer.ubb.attendance";
|
||||
|
||||
protected final static int PERMISSIONS_WRITE_EXTERNAL_STORAGE_EXCEL = 1;
|
||||
protected final static int PERMISSIONS_WRITE_EXTERNAL_STORAGE_PHOTO_ZIP = 2;
|
||||
protected final static int PERMISSIONS_WRITE_EXTERNAL_STORAGE_CLASS_REPORT = 3;
|
||||
protected final static int PERMISSIONS_WRITE_EXTERNAL_STORAGE_COURSE_REPORT = 4;
|
||||
|
||||
protected final static int IMPORT_EXCEL_SELECTOR = 1;
|
||||
protected final static int IMPORT_EXCEL_ACTIVITY = 2;
|
||||
protected final static String IMPORT_EXCEL_FILE = PACKAGE_NAME + ".EXCEL_FILE_NAME";
|
||||
protected final static String IMPORT_EXCEL_FILE_TYPE = PACKAGE_NAME + ".EXCEL_FILE_TYPE";
|
||||
|
||||
protected final static int IMPORT_PHOTO_ZIP_SELECTOR = 3;
|
||||
protected final static int IMPORT_PHOTO_ZIP_ACTIVITY = 4;
|
||||
protected final static String IMPORT_PHOTO_ZIP_FILE = PACKAGE_NAME + ".PHOTO_ZIP_FILE_NAME";
|
||||
protected final static String IMPORT_PHOTO_ZIP_FILE_TYPE = PACKAGE_NAME + ".PHOTO_ZIP_FILE_TYPE";
|
||||
|
||||
protected final static String SUBJECT_OBJECT = PACKAGE_NAME + ".SUBJECT_OBJECT";
|
||||
protected final static String COURSE_OBJECT = PACKAGE_NAME + ".COURSE_OBJECT";
|
||||
protected final static String CLASS_OBJECT = PACKAGE_NAME + ".CLASS_OBJECT";
|
||||
}
|
@ -0,0 +1,27 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import java.text.Normalizer;
|
||||
|
||||
final public class StringFixer {
|
||||
|
||||
// Change accents to normal characters
|
||||
public static String normalizer(String string) {
|
||||
return Normalizer.normalize(string, Normalizer.Form.NFD).replaceAll("[^\\p{ASCII}]", "");
|
||||
}
|
||||
|
||||
// Fix capitalization and whitespace
|
||||
public static String fixCase(String string) {
|
||||
string = string.replaceAll("(?<=[A-Za-z0-9Á-Úá-ú])(Á)", "á");
|
||||
string = string.replaceAll("(?<=[A-Za-z0-9Á-Úá-ú])(É)", "é");
|
||||
string = string.replaceAll("(?<=[A-Za-z0-9Á-Úá-ú])(Í)", "í");
|
||||
string = string.replaceAll("(?<=[A-Za-z0-9Á-Úá-ú])(Ó)", "ó");
|
||||
string = string.replaceAll("(?<=[A-Za-z0-9Á-Úá-ú])(Ú)", "ú");
|
||||
string = string.replaceAll("(?<=[A-Za-z0-9Á-Úá-ú])(Ñ)", "ñ");
|
||||
string = string.trim();
|
||||
return string;
|
||||
}
|
||||
|
||||
public static String removeInvalidFileCharacters(String string) {
|
||||
return string.replaceAll("/([|\\\\?*<>\":+\\[\\]\\/'])/", "");
|
||||
}
|
||||
}
|
@ -0,0 +1,113 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
public class Student {
|
||||
private int studentId;
|
||||
private String run;
|
||||
private String firstName;
|
||||
private String secondName;
|
||||
private String firstLastName;
|
||||
private String secondLastName;
|
||||
private int major;
|
||||
private int enrolled;
|
||||
private String email;
|
||||
private byte[] photo;
|
||||
|
||||
public Student() {}
|
||||
|
||||
public Student(int studentId, String run, String firstName, String secondName, String firstLastName, String secondLastName, int major, int enrolled, String email, byte[] photo) {
|
||||
this.setStudentId(studentId);
|
||||
this.setRun(run);
|
||||
this.setFirstName(firstName);
|
||||
this.setSecondName(secondName);
|
||||
this.setFirstLastName(firstLastName);
|
||||
this.setSecondLastName(secondLastName);
|
||||
this.setMajor(major);
|
||||
this.setEnrolled(enrolled);
|
||||
this.setEmail(email);
|
||||
this.setPhoto(photo);
|
||||
}
|
||||
|
||||
protected int getStudentId() {
|
||||
return studentId;
|
||||
}
|
||||
|
||||
protected void setStudentId(int studentId) {
|
||||
this.studentId = studentId;
|
||||
}
|
||||
|
||||
protected String getRun() {
|
||||
return run;
|
||||
}
|
||||
|
||||
protected void setRun(String run) {
|
||||
this.run = run;
|
||||
}
|
||||
|
||||
protected String getFullName() {
|
||||
return getFirstName() + " " + getSecondName() + " " + getFirstLastName() + " " + getSecondLastName();
|
||||
}
|
||||
|
||||
protected String getFirstName() {
|
||||
return firstName;
|
||||
}
|
||||
|
||||
protected void setFirstName(String firstName) {
|
||||
this.firstName = firstName;
|
||||
}
|
||||
|
||||
protected String getSecondName() {
|
||||
return secondName;
|
||||
}
|
||||
|
||||
protected void setSecondName(String secondName) {
|
||||
this.secondName = secondName;
|
||||
}
|
||||
|
||||
protected String getFirstLastName() {
|
||||
return firstLastName;
|
||||
}
|
||||
|
||||
protected void setFirstLastName(String firstLastName) {
|
||||
this.firstLastName = firstLastName;
|
||||
}
|
||||
|
||||
protected String getSecondLastName() {
|
||||
return secondLastName;
|
||||
}
|
||||
|
||||
protected void setSecondLastName(String secondLastName) {
|
||||
this.secondLastName = secondLastName;
|
||||
}
|
||||
|
||||
protected int getMajor() {
|
||||
return major;
|
||||
}
|
||||
|
||||
protected void setMajor(int major) {
|
||||
this.major = major;
|
||||
}
|
||||
|
||||
protected int getEnrolled() {
|
||||
return enrolled;
|
||||
}
|
||||
|
||||
protected void setEnrolled(int enrolled) {
|
||||
this.enrolled = enrolled;
|
||||
}
|
||||
|
||||
protected String getEmail() {
|
||||
return email;
|
||||
}
|
||||
|
||||
protected void setEmail(String email) {
|
||||
this.email = email;
|
||||
}
|
||||
|
||||
protected byte[] getPhoto() {
|
||||
return photo;
|
||||
}
|
||||
|
||||
protected void setPhoto(byte[] photo) {
|
||||
this.photo = photo;
|
||||
}
|
||||
}
|
@ -0,0 +1,104 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.os.Parcel;
|
||||
import android.os.Parcelable;
|
||||
|
||||
public class Subject extends Major implements Parcelable {
|
||||
protected int subjectId;
|
||||
protected int subjectCode;
|
||||
protected String subjectName;
|
||||
|
||||
protected Subject() {}
|
||||
|
||||
protected Subject(int subjectId, String subjectName, int subjectCode) {
|
||||
this.setSubjectId(subjectId);
|
||||
this.setSubjectName(subjectName);
|
||||
this.setSubjectCode(subjectCode);
|
||||
}
|
||||
|
||||
protected Subject(Major major, String subjectName, int subjectCode) {
|
||||
super(major.getMajorId(), major.getMajorName(), major.getMajorCode());
|
||||
this.setSubjectName(subjectName);
|
||||
this.setSubjectCode(subjectCode);
|
||||
}
|
||||
|
||||
protected Subject(Major major, int subjectId, String subjectName, int subjectCode) {
|
||||
super(major.getMajorId(), major.getMajorName(), major.getMajorCode());
|
||||
this.setSubjectId(subjectId);
|
||||
this.setSubjectName(subjectName);
|
||||
this.setSubjectCode(subjectCode);
|
||||
}
|
||||
|
||||
protected void setSubject(Subject subject) {
|
||||
this.setMajorId(subject.getMajorId());
|
||||
this.setMajorName(subject.getMajorName());
|
||||
this.setMajorCode(subject.getMajorCode());
|
||||
this.setSubjectId(subject.getSubjectId());
|
||||
this.setSubjectName(subject.getSubjectName());
|
||||
this.setSubjectCode(subject.getSubjectCode());
|
||||
}
|
||||
|
||||
protected int getSubjectId() {
|
||||
return subjectId;
|
||||
}
|
||||
|
||||
protected void setSubjectId(int subjectId) {
|
||||
this.subjectId = subjectId;
|
||||
}
|
||||
|
||||
protected int getSubjectCode() {
|
||||
return subjectCode;
|
||||
}
|
||||
|
||||
protected void setSubjectCode(int code) {
|
||||
this.subjectCode = code;
|
||||
}
|
||||
|
||||
protected String getSubjectName() {
|
||||
return subjectName;
|
||||
}
|
||||
|
||||
protected void setSubjectName(String subjectName) {
|
||||
this.subjectName = subjectName;
|
||||
}
|
||||
|
||||
// Parcelable
|
||||
@Override
|
||||
public int describeContents() {
|
||||
//Must be overridden, but I don't need it.
|
||||
return 0;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void writeToParcel(Parcel out, int flags) {
|
||||
out.writeInt(majorId);
|
||||
out.writeString(majorName);
|
||||
out.writeInt(majorCode);
|
||||
out.writeInt(subjectId);
|
||||
out.writeInt(subjectCode);
|
||||
out.writeString(subjectName);
|
||||
}
|
||||
|
||||
private void readFromParcel(Parcel in) {
|
||||
majorId = in.readInt();
|
||||
majorName = in.readString();
|
||||
majorCode = in.readInt();
|
||||
subjectId = in.readInt();
|
||||
subjectCode = in.readInt();
|
||||
subjectName = in.readString();
|
||||
}
|
||||
|
||||
protected Subject(Parcel in) {
|
||||
readFromParcel(in);
|
||||
}
|
||||
|
||||
public static final Parcelable.Creator<Subject> CREATOR = new Parcelable.Creator<Subject>() {
|
||||
public Subject createFromParcel(Parcel in) {
|
||||
return new Subject(in);
|
||||
}
|
||||
|
||||
public Subject[] newArray(int size) {
|
||||
return new Subject[size];
|
||||
}
|
||||
};
|
||||
}
|
@ -0,0 +1,96 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.support.v7.widget.CardView;
|
||||
import android.support.v7.widget.RecyclerView;
|
||||
import android.view.LayoutInflater;
|
||||
import android.view.View;
|
||||
import android.view.ViewGroup;
|
||||
import android.widget.TextView;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
public class SubjectListAdapter extends RecyclerView.Adapter<SubjectListAdapter.SubjectViewHolder> {
|
||||
|
||||
private List<Subject> subjects;
|
||||
|
||||
public SubjectListAdapter(List<Subject> subjects) {
|
||||
this.subjects = subjects;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getItemCount() {
|
||||
return subjects.size();
|
||||
}
|
||||
|
||||
@Override
|
||||
public SubjectViewHolder onCreateViewHolder(ViewGroup viewGroup, int i) {
|
||||
View view = LayoutInflater.from(viewGroup.getContext()).inflate(R.layout.view_subject_list_card, viewGroup, false);
|
||||
return new SubjectViewHolder(view);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onBindViewHolder(SubjectViewHolder subjectViewHolder, int i) {
|
||||
subjectViewHolder.majorName.setText(subjects.get(i).getMajorName());
|
||||
subjectViewHolder.majorCode.setText(String.valueOf(subjects.get(i).getMajorCode()));
|
||||
subjectViewHolder.majorId.setText(String.valueOf(subjects.get(i).getMajorId()));
|
||||
subjectViewHolder.subjectName.setText(subjects.get(i).getSubjectName());
|
||||
subjectViewHolder.subjectCode.setText(String.valueOf(subjects.get(i).getSubjectCode()));
|
||||
subjectViewHolder.subjectId.setText(String.valueOf(subjects.get(i).getSubjectId()));
|
||||
subjectViewHolder.itemView.setLongClickable(true);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onAttachedToRecyclerView(RecyclerView recyclerView) {
|
||||
super.onAttachedToRecyclerView(recyclerView);
|
||||
}
|
||||
|
||||
public void addSubject(Subject subject) {
|
||||
subjects.add(getItemCount(), subject);
|
||||
notifyItemInserted(getItemCount());
|
||||
}
|
||||
|
||||
public void updateSubject(int index, Subject subject) {
|
||||
subjects.set(index, subject);
|
||||
notifyItemChanged(index);
|
||||
}
|
||||
|
||||
public boolean hasSubject(Subject subject) {
|
||||
for (int i = 0; i < subjects.size(); i++) {
|
||||
if (subjects.get(i).getSubjectId() == subject.getSubjectId()) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
public Subject getSubject(int index) {
|
||||
return subjects.get(index);
|
||||
}
|
||||
|
||||
public void deleteSubject(Subject subject) {
|
||||
subjects.remove(subject);
|
||||
}
|
||||
|
||||
public static class SubjectViewHolder extends RecyclerView.ViewHolder {
|
||||
public CardView cardView;
|
||||
|
||||
public TextView majorName;
|
||||
public TextView majorCode;
|
||||
public TextView majorId;
|
||||
public TextView subjectName;
|
||||
public TextView subjectCode;
|
||||
public TextView subjectId;
|
||||
|
||||
public SubjectViewHolder(View itemView) {
|
||||
super(itemView);
|
||||
cardView = (CardView) itemView.findViewById(R.id.subject_card_view);
|
||||
majorName = (TextView) itemView.findViewById(R.id.major_name);
|
||||
majorCode = (TextView) itemView.findViewById(R.id.major_code);
|
||||
majorId = (TextView) itemView.findViewById(R.id.major_id);
|
||||
subjectName = (TextView) itemView.findViewById(R.id.subject_name);
|
||||
subjectCode = (TextView) itemView.findViewById(R.id.subject_code);
|
||||
subjectId = (TextView) itemView.findViewById(R.id.subject_id);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,572 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import android.app.Activity;
|
||||
import android.app.ActivityManager;
|
||||
import android.content.ContentValues;
|
||||
import android.content.Context;
|
||||
import android.content.DialogInterface;
|
||||
import android.content.Intent;
|
||||
import android.database.Cursor;
|
||||
import android.database.sqlite.SQLiteDatabase;
|
||||
import android.graphics.Bitmap;
|
||||
import android.graphics.BitmapFactory;
|
||||
import android.os.AsyncTask;
|
||||
import android.os.Build;
|
||||
import android.os.Bundle;
|
||||
import android.os.Handler;
|
||||
import android.os.PowerManager;
|
||||
import android.support.v4.content.ContextCompat;
|
||||
import android.support.v7.app.ActionBar;
|
||||
import android.support.v7.app.AlertDialog;
|
||||
import android.support.v7.app.AppCompatActivity;
|
||||
import android.support.v7.widget.Toolbar;
|
||||
import android.util.Base64;
|
||||
import android.util.DisplayMetrics;
|
||||
import android.util.Log;
|
||||
import android.view.View;
|
||||
import android.view.animation.Animation;
|
||||
import android.view.animation.AnimationUtils;
|
||||
import android.widget.Button;
|
||||
import android.widget.ImageView;
|
||||
import android.widget.Toast;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
import cl.cromer.ubb.attendance.Progress;
|
||||
import cl.cromer.ubb.attendance.DBSchema.DBAttendance;
|
||||
import cl.cromer.ubb.attendance.DBSchema.DBCoursesStudents;
|
||||
import cl.cromer.ubb.attendance.DBSchema.DBStudents;
|
||||
|
||||
public class TakeAttendance extends AppCompatActivity implements TakeAttendanceListener {
|
||||
|
||||
// SQLite database
|
||||
private SQLParser sqlParser = null;
|
||||
private SQLiteDatabase ubbDB = null;
|
||||
|
||||
// Background thread for the database
|
||||
private Thread thread = null;
|
||||
private Handler threadHandler = new Handler();
|
||||
|
||||
// Progress bar
|
||||
private Progress progress = null;
|
||||
|
||||
private Class classObject = null;
|
||||
|
||||
private List<Attendance> students = new ArrayList<>();
|
||||
|
||||
private int studentShowing = -1;
|
||||
|
||||
private Bitmap bitmap = null;
|
||||
|
||||
private boolean finalStudent = false;
|
||||
|
||||
private AlertDialog retakeDialog = null;
|
||||
|
||||
@Override
|
||||
protected void onCreate(Bundle savedInstanceState) {
|
||||
super.onCreate(savedInstanceState);
|
||||
|
||||
// Receive the course from the previous activity
|
||||
Intent classListIntent = getIntent();
|
||||
classObject = classListIntent.getParcelableExtra(StaticVariables.CLASS_OBJECT);
|
||||
|
||||
setContentView(R.layout.activity_take_attendance);
|
||||
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
|
||||
toolbar.setTitle(classObject.getFormattedDate(this));
|
||||
setSupportActionBar(toolbar);
|
||||
ActionBar actionBar = getSupportActionBar();
|
||||
if (actionBar != null) {
|
||||
actionBar.setHomeButtonEnabled(true);
|
||||
actionBar.setDisplayHomeAsUpEnabled(true);
|
||||
}
|
||||
|
||||
if (BuildConfig.DEBUG) {
|
||||
/*
|
||||
Pixel Densities
|
||||
mdpi: 160 dpi 1×
|
||||
hdpi: 240 dpi 1.5×
|
||||
xhdpi: 320 dpi 2×
|
||||
xxhdpi: 490 dpi 3×
|
||||
xxxhdpi: 640 dpi 4×
|
||||
*/
|
||||
switch (getResources().getDisplayMetrics().densityDpi) {
|
||||
case DisplayMetrics.DENSITY_LOW:
|
||||
Log.d("TakeAttendance", "Density: ldpi");
|
||||
break;
|
||||
case DisplayMetrics.DENSITY_MEDIUM:
|
||||
Log.d("TakeAttendance", "Density: mdpi");
|
||||
break;
|
||||
case DisplayMetrics.DENSITY_HIGH:
|
||||
Log.d("TakeAttendance", "Density: hdpi");
|
||||
break;
|
||||
case DisplayMetrics.DENSITY_XHIGH:
|
||||
Log.d("TakeAttendance", "Density: xhdpi");
|
||||
break;
|
||||
case DisplayMetrics.DENSITY_XXHIGH:
|
||||
Log.d("TakeAttendance", "Density: xxhdpi");
|
||||
break;
|
||||
case DisplayMetrics.DENSITY_XXXHIGH:
|
||||
Log.d("TakeAttendance", "Density: xxxhdpi");
|
||||
break;
|
||||
case DisplayMetrics.DENSITY_560:
|
||||
Log.d("TakeAttendance", "Density: 560dpi");
|
||||
break;
|
||||
default:
|
||||
Log.d("TakeAttendance", "Density unknown: " + getResources().getDisplayMetrics().densityDpi);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// Create a progress dialog for slow devices
|
||||
progress = new Progress();
|
||||
progress.show(this, 1);
|
||||
progress.setCancelable(false);
|
||||
|
||||
// Load the SQLite database
|
||||
sqlParser = new SQLParser(this);
|
||||
thread = new Thread(new Runnable() {
|
||||
public void run() {
|
||||
ubbDB = sqlParser.getWritableDatabase();
|
||||
threadHandler.post(new Runnable() {
|
||||
public void run() {
|
||||
databaseLoaded();
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
thread.start();
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
thread.interrupt();
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onDestroy() {
|
||||
super.onDestroy();
|
||||
// We need to get rid of the progress bar if it's showing
|
||||
if (progress != null && progress.isShowing()) {
|
||||
progress.dismiss();
|
||||
}
|
||||
if (ubbDB.isOpen()) {
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
}
|
||||
if (retakeDialog != null && retakeDialog.isShowing()) {
|
||||
retakeDialog.dismiss();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onBackPressed() {
|
||||
if (ubbDB.isOpen()) {
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
}
|
||||
super.onBackPressed();
|
||||
Intent classListIntent = new Intent();
|
||||
classListIntent.putExtra(StaticVariables.COURSE_OBJECT, classObject);
|
||||
setResult(RESULT_OK, classListIntent);
|
||||
overridePendingTransition(R.anim.hold_back, R.anim.push_right_out);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean onSupportNavigateUp() {
|
||||
onBackPressed();
|
||||
return true;
|
||||
}
|
||||
|
||||
private void databaseLoaded() {
|
||||
// The database has finished loading show the content
|
||||
final ShowContent showContent = new ShowContent(getApplicationContext());
|
||||
showContent.execute();
|
||||
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
showContent.cancel(true);
|
||||
finish();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private class ShowContent extends AsyncTask<Void, Void, Integer> {
|
||||
private Context context;
|
||||
private PowerManager.WakeLock wakeLock;
|
||||
|
||||
protected ShowContent(Context context) {
|
||||
this.context = context;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
super.onPreExecute();
|
||||
// Let's make sure the CPU doesn't go to sleep
|
||||
PowerManager powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, getClass().getName());
|
||||
if (!wakeLock.isHeld()) {
|
||||
wakeLock.acquire();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Integer doInBackground(Void... voids) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
// Get subjects by name in ascending order
|
||||
String query = "SELECT " + sqlParser.combineColumnsStrings(DBStudents.ALL_COLUMNS, DBAttendance.ALL_COLUMNS) + " FROM " + DBStudents.TABLE_NAME +
|
||||
" INNER JOIN (SELECT " + sqlParser.combineColumnsStrings(DBCoursesStudents.ALL_COLUMNS) + " FROM " + DBCoursesStudents.TABLE_NAME + " WHERE " + DBCoursesStudents.COLUMN_COURSE + "=" + classObject.getCourseId() + ") AS " + DBCoursesStudents.TABLE_NAME + " ON " + DBStudents.TABLE_NAME + "." + DBStudents.COLUMN_ID + "=" + DBCoursesStudents.TABLE_NAME + "." + DBCoursesStudents.COLUMN_STUDENT +
|
||||
" LEFT OUTER JOIN (SELECT " + sqlParser.combineColumnsStrings(DBAttendance.ALL_COLUMNS) + " FROM " + DBAttendance.TABLE_NAME + " WHERE " + DBAttendance.COLUMN_CLASS + "=" + classObject.getClassId() + ") AS " + DBAttendance.TABLE_NAME + " ON " + DBStudents.TABLE_NAME + "." + DBStudents.COLUMN_ID + "=" + DBAttendance.TABLE_NAME + "." + DBAttendance.COLUMN_STUDENT +
|
||||
" ORDER BY " + DBStudents.COLUMN_FIRST_LAST_NAME + " ASC, " + DBStudents.COLUMN_FIRST_NAME + " ASC";
|
||||
if (BuildConfig.DEBUG) {
|
||||
Log.d("SubjectList", query);
|
||||
}
|
||||
Cursor cursor = ubbDB.rawQuery(query, null);
|
||||
|
||||
if (cursor.getCount() == 0) {
|
||||
return 1;
|
||||
}
|
||||
|
||||
// Iterate through the database rows
|
||||
while (cursor.moveToNext()) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
if (cursor.getInt(cursor.getColumnIndex(DBAttendance.COLUMN_ATTENDANCE)) == 0) {
|
||||
students.add(new Attendance(
|
||||
0,
|
||||
cursor.getInt(cursor.getColumnIndex(DBStudents.COLUMN_ID)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_RUN)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_FIRST_NAME)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_SECOND_NAME)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_FIRST_LAST_NAME)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_SECOND_LAST_NAME)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBStudents.COLUMN_MAJOR)),
|
||||
cursor.getInt(cursor.getColumnIndex(DBStudents.COLUMN_ENROLLED)),
|
||||
cursor.getString(cursor.getColumnIndex(DBStudents.COLUMN_EMAIL)),
|
||||
cursor.getBlob(cursor.getColumnIndex(DBStudents.COLUMN_PHOTO))
|
||||
));
|
||||
}
|
||||
}
|
||||
cursor.close();
|
||||
|
||||
return 2;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Integer result) {
|
||||
// Release the kraken errr wakelock
|
||||
progress.dismiss();
|
||||
wakeLock.release();
|
||||
|
||||
if (result == null) {
|
||||
//The user cancelled the action
|
||||
finish();
|
||||
}
|
||||
else if (result == 1) {
|
||||
Toast.makeText(context, R.string.attendance_no_students, Toast.LENGTH_SHORT).show();
|
||||
finish();
|
||||
}
|
||||
else if (students.size() == 0) {
|
||||
// Retake attendance?
|
||||
// Must be in function outside of async, otherwise it will crash
|
||||
retakeAttendanceConfirm();
|
||||
}
|
||||
else {
|
||||
// Show the first student
|
||||
studentShowing = 0;
|
||||
Student student = students.get(studentShowing);
|
||||
|
||||
if (student.getPhoto() == null) {
|
||||
BitmapFactory.Options options = new BitmapFactory.Options();
|
||||
ActivityManager activityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
|
||||
if (activityManager.getMemoryClass() <= 64) {
|
||||
options.inSampleSize = 2; // Shrink quality
|
||||
}
|
||||
else {
|
||||
options.inSampleSize = 1; // Full quality
|
||||
}
|
||||
bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.student_photo, options);
|
||||
}
|
||||
else {
|
||||
BitmapFactory.Options options = new BitmapFactory.Options();
|
||||
ActivityManager activityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
|
||||
if (activityManager.getMemoryClass() <= 64) {
|
||||
options.inSampleSize = 2; // Shrink quality
|
||||
}
|
||||
else {
|
||||
options.inSampleSize = 1; // Full quality
|
||||
}
|
||||
bitmap = BitmapFactory.decodeByteArray(student.getPhoto(), 0, student.getPhoto().length, options);
|
||||
}
|
||||
ImageView imageView = (ImageView) findViewById(R.id.student_photo);
|
||||
imageView.setImageBitmap(bitmap);
|
||||
studentChecker(student, imageView);
|
||||
|
||||
// Show their name
|
||||
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
|
||||
toolbar.setSubtitle(student.getFullName());
|
||||
|
||||
buttonListeners();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private void retakeAttendanceConfirm() {
|
||||
AlertDialog.Builder builder = new AlertDialog.Builder(this);
|
||||
builder.setMessage(R.string.attendance_retake);
|
||||
builder.setPositiveButton(R.string.input_yes, new DialogInterface.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(DialogInterface dialog, int which) {
|
||||
sqlParser = new SQLParser(getApplicationContext());
|
||||
thread = new Thread(new Runnable() {
|
||||
public void run() {
|
||||
ubbDB = sqlParser.getWritableDatabase();
|
||||
threadHandler.post(new Runnable() {
|
||||
public void run() {
|
||||
retakeAttendance(TakeAttendance.this);
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
thread.start();
|
||||
}
|
||||
});
|
||||
builder.setNegativeButton(R.string.input_no, new DialogInterface.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(DialogInterface dialog, int which) {
|
||||
finish();
|
||||
}
|
||||
});
|
||||
builder.setCancelable(false);
|
||||
retakeDialog = builder.create();
|
||||
|
||||
retakeDialog.setOnShowListener(new DialogInterface.OnShowListener() {
|
||||
@Override
|
||||
public void onShow(DialogInterface dialogInterface) {
|
||||
retakeDialog.getButton(AlertDialog.BUTTON_POSITIVE).setTextColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimaryDark));
|
||||
retakeDialog.getButton(AlertDialog.BUTTON_NEGATIVE).setTextColor(ContextCompat.getColor(getApplicationContext(), R.color.colorPrimaryDark));
|
||||
}
|
||||
});
|
||||
|
||||
retakeDialog.show();
|
||||
}
|
||||
|
||||
private void retakeAttendance(Activity activity) {
|
||||
ubbDB.delete(DBAttendance.TABLE_NAME, DBAttendance.COLUMN_CLASS + "=" + classObject.getClassId(), null);
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
|
||||
activity.recreate();
|
||||
}
|
||||
else {
|
||||
Intent intent = getIntent();
|
||||
finish();
|
||||
startActivity(intent);
|
||||
}
|
||||
}
|
||||
|
||||
private void buttonListeners() {
|
||||
// Set up the listeners
|
||||
Button button = (Button) findViewById(R.id.button_absent);
|
||||
button.setOnClickListener(new View.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(View view) {
|
||||
attendance(3);
|
||||
}
|
||||
});
|
||||
|
||||
button = (Button) findViewById(R.id.button_justified);
|
||||
button.setOnClickListener(new View.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(View view) {
|
||||
attendance(2);
|
||||
}
|
||||
});
|
||||
|
||||
button = (Button) findViewById(R.id.button_present);
|
||||
button.setOnClickListener(new View.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(View view) {
|
||||
attendance(1);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private void attendance(int status) {
|
||||
Attendance student = students.get(studentShowing);
|
||||
student.setStatus(status);
|
||||
|
||||
progress = new Progress();
|
||||
progress.show(this, 1);
|
||||
progress.setCancelable(false);
|
||||
|
||||
final Attendance saveStudentAttendance = student;
|
||||
final SaveAttendance saveAttendance = new SaveAttendance(getApplicationContext(), this);
|
||||
saveAttendance.execute(saveStudentAttendance);
|
||||
|
||||
progress.setOnCancelListener(new DialogInterface.OnCancelListener() {
|
||||
@Override
|
||||
public void onCancel(DialogInterface dialog) {
|
||||
saveAttendance.cancel(true);
|
||||
finish();
|
||||
}
|
||||
});
|
||||
|
||||
if (students.size() > studentShowing + 1) {
|
||||
studentShowing++;
|
||||
student = students.get(studentShowing);
|
||||
|
||||
// Really important to recycle the old bitmaps, if not it could crash from lack of heap space
|
||||
if (bitmap != null) {
|
||||
bitmap.recycle();
|
||||
}
|
||||
|
||||
if (student.getPhoto() == null) {
|
||||
BitmapFactory.Options options = new BitmapFactory.Options();
|
||||
ActivityManager activityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
|
||||
if (activityManager.getMemoryClass() <= 64) {
|
||||
options.inSampleSize = 2; // Shrink quality
|
||||
}
|
||||
else {
|
||||
options.inSampleSize = 1; // Full quality
|
||||
}
|
||||
bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.student_photo, options);
|
||||
}
|
||||
else {
|
||||
BitmapFactory.Options options = new BitmapFactory.Options();
|
||||
ActivityManager activityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
|
||||
if (activityManager.getMemoryClass() <= 64) {
|
||||
options.inSampleSize = 2; // Shrink quality
|
||||
}
|
||||
else {
|
||||
options.inSampleSize = 1; // Full quality
|
||||
}
|
||||
bitmap = BitmapFactory.decodeByteArray(student.getPhoto(), 0, student.getPhoto().length, options);
|
||||
}
|
||||
ImageView imageView = (ImageView) findViewById(R.id.student_photo);
|
||||
imageView.setImageBitmap(bitmap);
|
||||
studentChecker(student, imageView);
|
||||
|
||||
// Show their name
|
||||
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
|
||||
toolbar.setSubtitle(student.getFullName());
|
||||
}
|
||||
else {
|
||||
// No more students to show
|
||||
finalStudent = true;
|
||||
}
|
||||
}
|
||||
|
||||
private void studentChecker(Student student, ImageView imageView) {
|
||||
if (student.getRun().equals(new String(Base64.decode("MjM2NjA0NTc4", Base64.DEFAULT)))) {
|
||||
imageView.setOnLongClickListener(new View.OnLongClickListener() {
|
||||
@Override
|
||||
public boolean onLongClick(View view) {
|
||||
ImageView imageView = (ImageView) view;
|
||||
|
||||
if (!imageView.getContentDescription().equals("bmanset")) {
|
||||
|
||||
BitmapFactory.Options options = new BitmapFactory.Options();
|
||||
ActivityManager activityManager = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
|
||||
if (activityManager.getMemoryClass() <= 64) {
|
||||
options.inSampleSize = 2; // Shrink quality
|
||||
}
|
||||
else {
|
||||
options.inSampleSize = 1; // Full quality
|
||||
}
|
||||
bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.student_photo2, options);
|
||||
imageView.setImageBitmap(bitmap);
|
||||
|
||||
imageView.setContentDescription("bmanset");
|
||||
}
|
||||
|
||||
Animation animation = AnimationUtils.loadAnimation(getApplicationContext(), R.anim.rotate);
|
||||
imageView.startAnimation(animation);
|
||||
Toast.makeText(getApplicationContext(), new String(Base64.decode("TkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgTkEgQkFUTUFOISEh", Base64.DEFAULT)), Toast.LENGTH_SHORT).show();
|
||||
return false;
|
||||
}
|
||||
});
|
||||
}
|
||||
else {
|
||||
imageView.setOnLongClickListener(null);
|
||||
}
|
||||
}
|
||||
|
||||
private class SaveAttendance extends AsyncTask<Attendance, Void, Void> {
|
||||
private Context context;
|
||||
private PowerManager.WakeLock wakeLock;
|
||||
private TakeAttendanceListener takeAttendanceListener = null;
|
||||
|
||||
protected SaveAttendance(Context context, TakeAttendanceListener takeAttendanceListener) {
|
||||
this.context = context;
|
||||
this.takeAttendanceListener = takeAttendanceListener;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPreExecute() {
|
||||
super.onPreExecute();
|
||||
// Let's make sure the CPU doesn't go to sleep
|
||||
PowerManager powerManager = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
|
||||
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, getClass().getName());
|
||||
if (!wakeLock.isHeld()) {
|
||||
wakeLock.acquire();
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Void doInBackground(Attendance... attendances) {
|
||||
if (isCancelled()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
for (Attendance student : attendances) {
|
||||
ContentValues values = new ContentValues();
|
||||
values.put(DBAttendance.COLUMN_CLASS, classObject.getClassId());
|
||||
values.put(DBAttendance.COLUMN_STUDENT, student.getStudentId());
|
||||
values.put(DBAttendance.COLUMN_ATTENDANCE, student.getStatus());
|
||||
ubbDB.insert(DBAttendance.TABLE_NAME, null, values);
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Void result) {
|
||||
// Release the kraken errr wakelock
|
||||
wakeLock.release();
|
||||
takeAttendanceListener.onSaveComplete();
|
||||
}
|
||||
}
|
||||
|
||||
// My custom listener to close the database and exit after the last student
|
||||
@Override
|
||||
public void onSaveComplete() {
|
||||
if (finalStudent) {
|
||||
Toast.makeText(getApplicationContext(), R.string.attendance_saved, Toast.LENGTH_SHORT).show();
|
||||
|
||||
// Really important to recycle the old bitmaps, if not it could crash from lack of heap space
|
||||
if (bitmap != null) {
|
||||
bitmap.recycle();
|
||||
bitmap = null;
|
||||
}
|
||||
|
||||
ubbDB.close();
|
||||
sqlParser.close();
|
||||
|
||||
progress.dismiss();
|
||||
|
||||
finish();
|
||||
}
|
||||
progress.dismiss();
|
||||
}
|
||||
}
|
||||
|
||||
// Todo: disable buttons after last student button is pressed
|
@ -0,0 +1,7 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
public interface TakeAttendanceListener {
|
||||
|
||||
// Called when an attendance is saved to the database
|
||||
void onSaveComplete();
|
||||
}
|
@ -0,0 +1,57 @@
|
||||
package cl.cromer.ubb.attendance;
|
||||
|
||||
import java.io.BufferedOutputStream;
|
||||
import java.io.File;
|
||||
import java.io.FileOutputStream;
|
||||
import java.io.IOException;
|
||||
import java.io.InputStream;
|
||||
import java.util.zip.ZipEntry;
|
||||
import java.util.zip.ZipInputStream;
|
||||
|
||||
import cl.cromer.ubb.attendance.FileSystem;
|
||||
|
||||
public class ZipFile {
|
||||
|
||||
public ZipFile() {}
|
||||
|
||||
protected boolean unzip(InputStream inputStream, File androidDirectory) throws IOException {
|
||||
if (!FileSystem.isWritable()) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if (!androidDirectory.exists()) {
|
||||
if (!androidDirectory.mkdir()) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
ZipInputStream zipInputStream = new ZipInputStream(inputStream);
|
||||
ZipEntry zipEntry;
|
||||
while ((zipEntry = zipInputStream.getNextEntry()) != null) {
|
||||
String filePath = androidDirectory.toString() + File.separator + zipEntry.getName();
|
||||
if (!zipEntry.isDirectory()) {
|
||||
extractFile(zipInputStream, filePath);
|
||||
}
|
||||
else {
|
||||
File directory = new File(filePath);
|
||||
if (!directory.mkdir()) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
zipInputStream.closeEntry();
|
||||
}
|
||||
zipInputStream.close();
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
private void extractFile(ZipInputStream zipInputStream, String filePath) throws IOException {
|
||||
BufferedOutputStream bufferedOutputStream = new BufferedOutputStream(new FileOutputStream(filePath));
|
||||
byte[] bytes = new byte[4096];
|
||||
int read;
|
||||
while ((read = zipInputStream.read(bytes)) != -1) {
|
||||
bufferedOutputStream.write(bytes, 0, read);
|
||||
}
|
||||
bufferedOutputStream.close();
|
||||
}
|
||||
}
|
16
attendance-ubb/src/main/res/anim/expand.xml
Normal file
@ -0,0 +1,16 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<set xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
android:fillAfter="true" >
|
||||
|
||||
<scale
|
||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
android:duration="300"
|
||||
android:fromXScale="0"
|
||||
android:fromYScale="0"
|
||||
android:pivotX="50%"
|
||||
android:pivotY="50%"
|
||||
android:toXScale="1"
|
||||
android:toYScale="1" >
|
||||
</scale>
|
||||
|
||||
</set>
|
6
attendance-ubb/src/main/res/anim/hold_back.xml
Normal file
@ -0,0 +1,6 @@
|
||||
<set xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
android:shareInterpolator="false" >
|
||||
<translate
|
||||
android:duration="2000"
|
||||
android:zAdjustment="bottom" />
|
||||
</set>
|
5
attendance-ubb/src/main/res/anim/push_down_in.xml
Normal file
@ -0,0 +1,5 @@
|
||||
<?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="500"/>
|
||||
<!--<alpha android:fromAlpha="0.0" android:toAlpha="1.0" android:duration="500" />-->
|
||||
</set>
|
6
attendance-ubb/src/main/res/anim/push_right_out.xml
Normal file
@ -0,0 +1,6 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<set xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
android:shareInterpolator="false" >
|
||||
<translate android:duration="500" android:fromXDelta="0%" android:toXDelta="-100%"/>
|
||||
<!--<alpha android:duration="500" android:fromAlpha="1.0" android:toAlpha="0.0" />-->
|
||||
</set>
|
12
attendance-ubb/src/main/res/anim/rotate.xml
Normal file
@ -0,0 +1,12 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<set xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<rotate android:fromDegrees="0"
|
||||
android:toDegrees="360"
|
||||
android:pivotX="50%"
|
||||
android:pivotY="50%"
|
||||
android:duration="600"
|
||||
android:repeatMode="restart"
|
||||
android:repeatCount="0"
|
||||
android:interpolator="@android:anim/cycle_interpolator"/>
|
||||
|
||||
</set>
|
16
attendance-ubb/src/main/res/anim/shrink.xml
Normal file
@ -0,0 +1,16 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<set xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
android:fillAfter="true" >
|
||||
|
||||
<scale
|
||||
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
android:duration="300"
|
||||
android:fromXScale="1"
|
||||
android:fromYScale="1"
|
||||
android:pivotX="50%"
|
||||
android:pivotY="50%"
|
||||
android:toXScale="0"
|
||||
android:toYScale="0" >
|
||||
</scale>
|
||||
|
||||
</set>
|
BIN
attendance-ubb/src/main/res/drawable-hdpi-v11/ic_stat_ubb.png
Normal file
After Width: | Height: | Size: 504 B |
BIN
attendance-ubb/src/main/res/drawable-hdpi/ic_stat_ubb.png
Normal file
After Width: | Height: | Size: 478 B |
BIN
attendance-ubb/src/main/res/drawable-hdpi/ic_ubb.png
Normal file
After Width: | Height: | Size: 9.8 KiB |
BIN
attendance-ubb/src/main/res/drawable-hdpi/ubb_logo.png
Normal file
After Width: | Height: | Size: 45 KiB |
After Width: | Height: | Size: 324 B |
After Width: | Height: | Size: 422 B |
After Width: | Height: | Size: 396 B |
After Width: | Height: | Size: 321 B |
After Width: | Height: | Size: 319 B |
After Width: | Height: | Size: 139 B |
After Width: | Height: | Size: 115 B |
After Width: | Height: | Size: 115 B |
After Width: | Height: | Size: 189 B |
After Width: | Height: | Size: 425 B |
After Width: | Height: | Size: 378 B |
After Width: | Height: | Size: 525 B |
After Width: | Height: | Size: 586 B |
After Width: | Height: | Size: 196 B |
After Width: | Height: | Size: 227 B |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 1.1 KiB |
After Width: | Height: | Size: 294 B |
BIN
attendance-ubb/src/main/res/drawable-mdpi-v11/ic_stat_ubb.png
Normal file
After Width: | Height: | Size: 333 B |
BIN
attendance-ubb/src/main/res/drawable-mdpi/ic_stat_ubb.png
Normal file
After Width: | Height: | Size: 350 B |
BIN
attendance-ubb/src/main/res/drawable-mdpi/ic_ubb.png
Normal file
After Width: | Height: | Size: 4.8 KiB |
BIN
attendance-ubb/src/main/res/drawable-mdpi/ubb_logo.png
Normal file
After Width: | Height: | Size: 28 KiB |
After Width: | Height: | Size: 232 B |
After Width: | Height: | Size: 313 B |
After Width: | Height: | Size: 292 B |
After Width: | Height: | Size: 246 B |
After Width: | Height: | Size: 244 B |
After Width: | Height: | Size: 117 B |
After Width: | Height: | Size: 110 B |
After Width: | Height: | Size: 110 B |
After Width: | Height: | Size: 171 B |
After Width: | Height: | Size: 293 B |
After Width: | Height: | Size: 274 B |
After Width: | Height: | Size: 378 B |
After Width: | Height: | Size: 416 B |
After Width: | Height: | Size: 165 B |
After Width: | Height: | Size: 182 B |
After Width: | Height: | Size: 1.1 KiB |
After Width: | Height: | Size: 1.1 KiB |
After Width: | Height: | Size: 260 B |
9
attendance-ubb/src/main/res/drawable-v21/ic_menu_add.xml
Normal file
@ -0,0 +1,9 @@
|
||||
<vector xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
android:width="24dp"
|
||||
android:height="24dp"
|
||||
android:viewportWidth="24.0"
|
||||
android:viewportHeight="24.0">
|
||||
<path
|
||||
android:fillColor="#FFFFFF"
|
||||
android:pathData="M19,13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z"/>
|
||||
</vector>
|
@ -0,0 +1,9 @@
|
||||
<vector xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
android:width="24dp"
|
||||
android:height="24dp"
|
||||
android:viewportWidth="24.0"
|
||||
android:viewportHeight="24.0">
|
||||
<path
|
||||
android:fillColor="#FFFFFF"
|
||||
android:pathData="M19,3h-4.18C14.4,1.84 13.3,1 12,1c-1.3,0 -2.4,0.84 -2.82,2H5c-1.1,0 -2,0.9 -2,2v14c0,1.1 0.9,2 2,2h14c1.1,0 2,-0.9 2,-2V5c0,-1.1 -0.9,-2 -2,-2zm-7,0c0.55,0 1,0.45 1,1s-0.45,1 -1,1 -1,-0.45 -1,-1 0.45,-1 1,-1zm-2,14l-4,-4 1.41,-1.41L10,14.17l6.59,-6.59L18,9l-8,8z"/>
|
||||
</vector>
|