activityContext;
- private SyncCallback callback;
- private Exception e = null;
- private String apiResponse = null;
- private int profileId;
- private ProfileFull profile;
-
- public ProcessAsyncTask(App app, Context activityContext, SyncCallback callback, int profileId, ProfileFull profile) {
- //d(TAG, "Thread/ProcessAsyncTask/constructor/"+Thread.currentThread().getName());
- this.app = app;
- this.activityContext = new WeakReference<>(activityContext);
- this.callback = callback;
- this.profileId = profileId;
- this.profile = profile;
- }
-
- @Override
- protected Integer doInBackground(Void... voids) {
- Context activityContext = this.activityContext.get();
- //d(TAG, "Thread/ProcessAsyncTask/doInBackground/"+Thread.currentThread().getName());
- try {
-
- // UPDATE FCM TOKEN IF EMPTY
- if (app.appConfig.fcmToken == null || app.appConfig.fcmToken.equals("")) {
- FirebaseInstanceId.getInstance().getInstanceId().addOnSuccessListener(instanceIdResult -> {
- app.appConfig.fcmToken = instanceIdResult.getToken();
- app.appConfig.savePending = true;
- });
- }
-
- callback.onProgress(1);
- if (profile.getSyncNotifications()) {
- new Handler(activityContext.getMainLooper()).post(() -> {
- callback.onActionStarted(R.string.sync_action_creating_notifications);
- });
-
- for (LessonFull change : app.db.lessonChangeDao().getNotNotifiedNow(profileId)) {
- String text = app.getContext().getString(R.string.notification_lesson_change_format, change.changeTypeStr(app.getContext()), change.lessonDate == null ? "" : change.lessonDate.getFormattedString(), change.subjectLongName);
- app.notifier.add(new Notification(app.getContext(), text)
- .withProfileData(profile.getId(), profile.getName())
- .withType(Notification.TYPE_TIMETABLE_LESSON_CHANGE)
- .withFragmentRedirect(MainActivity.DRAWER_ITEM_TIMETABLE)
- .withLongExtra("timetableDate", change.lessonDate.getValue())
- .withAddedDate(change.addedDate)
- );
- }
- for (EventFull event : app.db.eventDao().getNotNotifiedNow(profileId)) {
- String text;
- if (event.type == TYPE_HOMEWORK)
- text = app.getContext().getString(R.string.notification_homework_format, ns(app.getString(R.string.notification_event_no_subject), event.subjectLongName), event.eventDate.getFormattedString());
- else
- text = app.getContext().getString(R.string.notification_event_format, event.typeName, event.eventDate.getFormattedString(), ns(app.getString(R.string.notification_event_no_subject), event.subjectLongName));
- app.notifier.add(new Notification(app.getContext(), text)
- .withProfileData(profile.getId(), profile.getName())
- .withType(event.type == TYPE_HOMEWORK ? Notification.TYPE_NEW_HOMEWORK : Notification.TYPE_NEW_EVENT)
- .withFragmentRedirect(event.type == TYPE_HOMEWORK ? MainActivity.DRAWER_ITEM_HOMEWORK : MainActivity.DRAWER_ITEM_AGENDA)
- .withLongExtra("eventId", event.id)
- .withLongExtra("eventDate", event.eventDate.getValue())
- .withAddedDate(event.addedDate)
- );
- // student's rights abuse - disabled, because this was useless
- /*if (!event.addedManually && event.type == RegisterEvent.TYPE_EXAM && event.eventDate.combineWith(event.startTime) - event.addedDate < 7 * 24 * 60 * 60 * 1000) {
- text = app.getContext().getString(R.string.notification_abuse_format, event.typeString(app, app.profile), event.subjectLongName, event.eventDate.getFormattedString());
- app.notifier.add(new Notification(app.getContext(), text)
- .withProfileData(profile.id, profile.name)
- .withType(Notification.TYPE_GENERAL)
- .withFragmentRedirect(MainActivity.DRAWER_ITEM_NOTIFICATIONS)
- );
- }*/
- }
-
- Date today = Date.getToday();
- int todayValue = today.getValue();
- profile.setCurrentSemester(profile.dateToSemester(today));
-
- for (GradeFull grade : app.db.gradeDao().getNotNotifiedNow(profileId)) {
- String gradeName = grade.name;
- if (grade.type == TYPE_SEMESTER1_PROPOSED
- || grade.type == TYPE_SEMESTER2_PROPOSED) {
- gradeName = (app.getString(R.string.grade_semester_proposed_format_2, grade.name));
- } else if (grade.type == TYPE_SEMESTER1_FINAL
- || grade.type == TYPE_SEMESTER2_FINAL) {
- gradeName = (app.getString(R.string.grade_semester_final_format_2, grade.name));
- } else if (grade.type == TYPE_YEAR_PROPOSED) {
- gradeName = (app.getString(R.string.grade_year_proposed_format_2, grade.name));
- } else if (grade.type == TYPE_YEAR_FINAL) {
- gradeName = (app.getString(R.string.grade_year_final_format_2, grade.name));
- }
- String text = app.getContext().getString(R.string.notification_grade_format, gradeName, grade.subjectLongName);
- app.notifier.add(new Notification(app.getContext(), text)
- .withProfileData(profile.getId(), profile.getName())
- .withType(Notification.TYPE_NEW_GRADE)
- .withFragmentRedirect(MainActivity.DRAWER_ITEM_GRADES)
- .withLongExtra("gradesSubjectId", grade.subjectId)
- .withAddedDate(grade.addedDate)
- );
- }
- for (NoticeFull notice : app.db.noticeDao().getNotNotifiedNow(profileId)) {
- String noticeTypeStr = (notice.type == Notice.TYPE_POSITIVE ? app.getString(R.string.notification_notice_praise) : (notice.type == Notice.TYPE_NEGATIVE ? app.getString(R.string.notification_notice_warning) : app.getString(R.string.notification_notice_new)));
- String text = app.getContext().getString(R.string.notification_notice_format, noticeTypeStr, notice.teacherFullName, Date.fromMillis(notice.addedDate).getFormattedString());
- app.notifier.add(new Notification(app.getContext(), text)
- .withProfileData(profile.getId(), profile.getName())
- .withType(Notification.TYPE_NEW_NOTICE)
- .withFragmentRedirect(MainActivity.DRAWER_ITEM_BEHAVIOUR)
- .withLongExtra("noticeId", notice.id)
- .withAddedDate(notice.addedDate)
- );
- }
- for (AttendanceFull attendance : app.db.attendanceDao().getNotNotifiedNow(profileId)) {
- String attendanceTypeStr = app.getString(R.string.notification_type_attendance);
- switch (attendance.type) {
- case Attendance.TYPE_ABSENT:
- attendanceTypeStr = app.getString(R.string.notification_absence);
- break;
- case Attendance.TYPE_ABSENT_EXCUSED:
- attendanceTypeStr = app.getString(R.string.notification_absence_excused);
- break;
- case Attendance.TYPE_BELATED:
- attendanceTypeStr = app.getString(R.string.notification_belated);
- break;
- case Attendance.TYPE_BELATED_EXCUSED:
- attendanceTypeStr = app.getString(R.string.notification_belated_excused);
- break;
- case Attendance.TYPE_RELEASED:
- attendanceTypeStr = app.getString(R.string.notification_release);
- break;
- }
- String text = app.getContext().getString(R.string.notification_attendance_format, attendanceTypeStr, attendance.subjectLongName, attendance.lessonDate.getFormattedString());
- app.notifier.add(new Notification(app.getContext(), text)
- .withProfileData(profile.getId(), profile.getName())
- .withType(Notification.TYPE_NEW_ATTENDANCE)
- .withFragmentRedirect(MainActivity.DRAWER_ITEM_ATTENDANCE)
- .withLongExtra("attendanceId", attendance.id)
- .withAddedDate(attendance.addedDate)
- );
- }
- for (AnnouncementFull announcement : app.db.announcementDao().getNotNotifiedNow(profileId)) {
- String text = app.getContext().getString(R.string.notification_announcement_format, announcement.subject);
- app.notifier.add(new Notification(app.getContext(), text)
- .withProfileData(profile.getId(), profile.getName())
- .withType(Notification.TYPE_NEW_ANNOUNCEMENT)
- .withFragmentRedirect(MainActivity.DRAWER_ITEM_ANNOUNCEMENTS)
- .withLongExtra("announcementId", announcement.id)
- .withAddedDate(announcement.addedDate)
- );
- }
- for (MessageFull message : app.db.messageDao().getReceivedNotNotifiedNow(profileId)) {
- String text = app.getContext().getString(R.string.notification_message_format, message.senderFullName, message.subject);
- app.notifier.add(new Notification(app.getContext(), text)
- .withProfileData(profile.getId(), profile.getName())
- .withType(Notification.TYPE_NEW_MESSAGE)
- .withFragmentRedirect(MainActivity.DRAWER_ITEM_MESSAGES)
- .withLongExtra("messageType", Message.TYPE_RECEIVED)
- .withLongExtra("messageId", message.id)
- .withAddedDate(message.addedDate)
- );
- }
-
- if (profile.getLuckyNumber() != oldLuckyNumber
- && profile.getLuckyNumber() != -1
- && profile.getLuckyNumberDate() != null
- && profile.getLuckyNumberDate().getValue() >= todayValue) {
- String text;
- if (profile.getLuckyNumberDate().getValue() == todayValue) { // LN for today
- text = app.getString((profile.getStudentNumber() != -1 && profile.getStudentNumber() == profile.getLuckyNumber() ? R.string.notification_lucky_number_yours_format : R.string.notification_lucky_number_format), profile.getLuckyNumber());
- } else if (profile.getLuckyNumberDate().getValue() == todayValue + 1) { // LN for tomorrow
- text = app.getString((profile.getStudentNumber() != -1 && profile.getStudentNumber() == profile.getLuckyNumber() ? R.string.notification_lucky_number_yours_tomorrow_format : R.string.notification_lucky_number_tomorrow_format), profile.getLuckyNumber());
- } else { // LN for later
- text = app.getString((profile.getStudentNumber() != -1 && profile.getStudentNumber() == profile.getLuckyNumber() ? R.string.notification_lucky_number_yours_later_format : R.string.notification_lucky_number_later_format), profile.getLuckyNumberDate().getFormattedString(), profile.getLuckyNumber());
- }
- app.notifier.add(new Notification(app.getContext(), text)
- .withProfileData(profile.getId(), profile.getName())
- .withType(Notification.TYPE_LUCKY_NUMBER)
- .withFragmentRedirect(MainActivity.DRAWER_ITEM_HOME)
- );
- oldLuckyNumber = profile.getLuckyNumber();
- }
- }
-
-
- app.db.metadataDao().setAllNotified(profileId, true);
- callback.onProgress(1);
-
- // SEND WEB PUSH, if registration allowed
- // otherwise, UNREGISTER THE USER
- if (profile.getRegistration() == REGISTRATION_ENABLED) {
- new Handler(activityContext.getMainLooper()).post(() -> {
- callback.onActionStarted(R.string.sync_action_syncing_shared_events);
- });
- //if (profile.registrationUsername == null || profile.registrationUsername.equals("")) {
- //}
- ServerRequest syncRequest = new ServerRequest(app, app.requestScheme + APP_URL + "main.php?sync", "Edziennik/REG", profile);
-
- if (registerEmpty) {
- syncRequest.setBodyParameter("first_run", "true");
- }
-
- // ALSO SEND NEW DATA TO BROWSER *excluding* all Shared Events !!!
- // because they will be sent by the server, as soon as it's shared, by FCM
-
- if (app.appConfig.webPushEnabled) {
- int position = 0;
- for (Notification notification : app.appConfig.notifications) {
- //Log.d(TAG, notification.text);
- if (!notification.notified) {
- if (notification.type != Notification.TYPE_NEW_SHARED_EVENT
- && notification.type != Notification.TYPE_SERVER_MESSAGE
- && notification.type != Notification.TYPE_NEW_SHARED_HOMEWORK) // these are automatically sent to the browser by the server
- {
- //Log.d(TAG, "Adding notify[" + position + "]");
- syncRequest.setBodyParameter("notify[" + position + "][type]", Integer.toString(notification.type));
- syncRequest.setBodyParameter("notify[" + position + "][title]", notification.title);
- syncRequest.setBodyParameter("notify[" + position + "][text]", notification.text);
- position++;
- }
- }
- }
- }
-
- callback.onProgress(1);
-
- if (app.appConfig.webPushEnabled || profile.getEnableSharedEvents()) {
- JsonObject result = syncRequest.runSync();
- callback.onProgress(1);
- //Log.d(TAG, "Executed request");
- if (result == null) {
- return AppError.CODE_APP_SERVER_ERROR;
- }
- apiResponse = result.toString();
- if (!result.get("success").getAsString().equals("true")) {
- return AppError.CODE_APP_SERVER_ERROR;
- }
- // HERE PROCESS ALL THE RECEIVED EVENTS
- // add them to the profile and create appropriate notifications
- for (JsonElement jEventEl : result.getAsJsonArray("events")) {
- JsonObject jEvent = jEventEl.getAsJsonObject();
- String teamCode = jEvent.get("team").getAsString();
- //d(TAG, "An event is there! "+jEvent.toString());
- // get the target Team from teamCode
- Team team = app.db.teamDao().getByCodeNow(profile.getId(), teamCode);
- if (team != null) {
- //d(TAG, "The target team is "+team.name+", ID "+team.id);
- // create the event from Json. Add the missing teamId and !!profileId!!
- Event event = app.gson.fromJson(jEvent.toString(), Event.class);
- // proguard. disable for Event.class
- if (event.eventDate == null) {
- apiResponse += "\n\nEventDate == null\n" + jEvent.toString();
- throw new Exception("null eventDate");
- }
- event.profileId = profile.getId();
- event.teamId = team.id;
- event.addedManually = true;
- //d(TAG, "Created the event! "+event);
-
- if (event.sharedBy != null && event.sharedBy.equals(profile.getUsernameId())) {
- //d(TAG, "Shared by self! Changing name");
- event.sharedBy = "self";
- event.sharedByName = profile.getStudentNameLong();
- }
-
- EventType type = app.db.eventTypeDao().getByIdNow(profileId, event.type);
-
- //d(TAG, "Finishing adding event "+event);
- app.db.eventDao().add(event);
- Metadata metadata = new Metadata(profile.getId(), event.type == TYPE_HOMEWORK ? Metadata.TYPE_HOMEWORK : Metadata.TYPE_EVENT, event.id, registerEmpty, true, jEvent.get("addedDate").getAsLong());
- long metadataId = app.db.metadataDao().add(metadata);
- if (metadataId != -1 && !registerEmpty) {
- app.notifier.add(new Notification(app.getContext(), app.getString(R.string.notification_shared_event_format, event.sharedByName, type != null ? type.name : "wydarzenie", event.eventDate == null ? "nieznana data" : event.eventDate.getFormattedString(), event.topic))
- .withProfileData(profile.getId(), profile.getName())
- .withType(event.type == TYPE_HOMEWORK ? Notification.TYPE_NEW_SHARED_HOMEWORK : Notification.TYPE_NEW_SHARED_EVENT)
- .withFragmentRedirect(event.type == TYPE_HOMEWORK ? MainActivity.DRAWER_ITEM_HOMEWORK : MainActivity.DRAWER_ITEM_AGENDA)
- .withLongExtra("eventDate", event.eventDate.getValue())
- );
- }
- }
- }
- callback.onProgress(5);
- return CODE_OK;
- } else {
- callback.onProgress(6);
- return CODE_OK;
- }
- } else {
- // the user does not want to be registered
- callback.onProgress(7);
- return CODE_OK;
- }
- } catch (Exception e) {
- e.printStackTrace();
- this.e = e;
- return null;
- }
- //return null;
- }
-
- @Override
- protected void onPostExecute(Integer errorCode) {
- //d(TAG, "Thread/ProcessAsyncTask/onPostExecute/"+Thread.currentThread().getName());
- Context activityContext = this.activityContext.get();
- app.profileSaveFull(profile);
- if (app.profile != null && profile.getId() == app.profile.getId()) {
- app.profile = profile;
- }
- if (errorCode == null) {
- // this means an Exception was thrown
- callback.onError(activityContext, new AppError(TAG, 513, CODE_OTHER, e, apiResponse));
- return;
- }
- //Log.d(TAG, "Finishing");
-
-
- callback.onProgress(1);
-
- if (errorCode == CODE_OK)
- callback.onSuccess(activityContext, profile);
- else {
- try {
- // oh that's useless
- throw new RuntimeException(stringErrorCode(app, errorCode, ""));
- } catch (Exception e) {
- callback.onError(activityContext, new AppError(TAG, 528, errorCode, e, (String) null));
- }
- }
- super.onPostExecute(errorCode);
- }
- }
-
- public void notifyAndReload() {
- app.notifier.postAll(null);
- app.saveConfig();
- SyncJob.schedule(app);
- Intent i = new Intent(Intent.ACTION_MAIN)
- .putExtra("reloadProfileId", -1);
- app.sendBroadcast(i);
-
- Intent intent = new Intent(app.getContext(), WidgetTimetable.class);
- intent.setAction(AppWidgetManager.ACTION_APPWIDGET_UPDATE);
- int[] ids = AppWidgetManager.getInstance(app).getAppWidgetIds(new ComponentName(app, WidgetTimetable.class));
- intent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_IDS, ids);
- app.sendBroadcast(intent);
-
- intent = new Intent(app.getContext(), WidgetNotifications.class);
- intent.setAction(AppWidgetManager.ACTION_APPWIDGET_UPDATE);
- ids = AppWidgetManager.getInstance(app).getAppWidgetIds(new ComponentName(app, WidgetNotifications.class));
- intent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_IDS, ids);
- app.sendBroadcast(intent);
-
- intent = new Intent(app.getContext(), WidgetLuckyNumber.class);
- intent.setAction(AppWidgetManager.ACTION_APPWIDGET_UPDATE);
- ids = AppWidgetManager.getInstance(app).getAppWidgetIds(new ComponentName(app, WidgetLuckyNumber.class));
- intent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_IDS, ids);
- app.sendBroadcast(intent);
- }
-
- /* _____
- / ____|
- | (___ _ _ _ __ ___
- \___ \| | | | '_ \ / __|
- ____) | |_| | | | | (__
- |_____/ \__, |_| |_|\___|
- __/ |
- |__*/
- // DataCallbacks that are *not* in Edziennik.sync need to be executed on the main thread.
- // EdziennikInterface.sync is executed on a worker thread
- // in Edziennik.sync/newCallback methods are called on a worker thread
-
- // callback passed to Edziennik.sync is executed on the main thread
- // thus, callback which is in guiSync is also on the main thread
-
- /**
- * Sync all Edziennik data.
- * Used in services, login form and {@code guiSync}
- *
- * May be ran on worker thread.
- * {@link EdziennikInterface}.sync is ran always on worker thread.
- * Every callback is ran on the UI thread.
- *
- * @param app
- * @param activityContext
- * @param callback
- * @param profileId
- */
- public void sync(@NonNull App app, @NonNull Context activityContext, @NonNull SyncCallback callback, int profileId) {
- sync(app, activityContext, callback, profileId, (int[])null);
- }
- public void sync(@NonNull App app, @NonNull Context activityContext, @NonNull SyncCallback callback, int profileId, @Nullable int ... featureList) {
- // empty: no unread notifications, all shared events (current+past)
- // only if there is no data, and we are not logged in yet
- SyncCallback newCallback = new SyncCallback() {
- @Override
- public void onLoginFirst(List profileList, LoginStore loginStore) {
- new Handler(activityContext.getMainLooper()).post(() -> {
- callback.onLoginFirst(profileList, loginStore);
- });
- }
-
- @Override
- public void onSuccess(Context activityContext, ProfileFull profileFull) {
- new Handler(activityContext.getMainLooper()).post(() -> {
- new ProcessAsyncTask(app, activityContext, callback, profileId, profileFull).execute();
- });
- }
-
- @Override
- public void onError(Context activityContext, AppError error) {
- new Handler(activityContext.getMainLooper()).post(() -> {
- callback.onError(activityContext, error);
- });
- }
-
- @Override
- public void onProgress(int progressStep) {
- new Handler(activityContext.getMainLooper()).post(() -> {
- callback.onProgress(progressStep);
- });
- }
-
- @Override
- public void onActionStarted(int stringResId) {
- new Handler(activityContext.getMainLooper()).post(() -> {
- callback.onActionStarted(stringResId);
- });
- }
- };
- AsyncTask.execute(() -> {
- ProfileFull profile = app.db.profileDao().getByIdNow(profileId);
- if (profile != null) {
-
- if (profile.getArchived()) {
- newCallback.onError(activityContext, new AppError(TAG, 678, CODE_PROFILE_ARCHIVED, profile.getName()));
- return;
- }
- else if (profile.getDateYearEnd() != null && Date.getToday().getValue() >= profile.getDateYearEnd().getValue()) {
- profile.setArchived(true);
- app.notifier.add(new Notification(app.getContext(), app.getString(R.string.profile_auto_archiving_format, profile.getName(), profile.getDateYearEnd().getFormattedString()))
- .withProfileData(profile.getId(), profile.getName())
- .withType(Notification.TYPE_AUTO_ARCHIVING)
- .withFragmentRedirect(DRAWER_ITEM_HOME)
- .withLongExtra("autoArchiving", 1L)
- );
- app.notifier.postAll(null);
- app.db.profileDao().add(profile);
- if (App.profileId == profile.getId()) {
- app.profile.setArchived(true);
- }
- newCallback.onSuccess(activityContext, profile);
- return;
- }
-
- registerEmpty = profile.getEmpty();
- oldLuckyNumber = profile.getLuckyNumber();
- getApi(app, profile.getLoginStoreType()).syncFeature(activityContext, newCallback, profile, featureList);
- } else {
- new Handler(activityContext.getMainLooper()).post(() -> callback.onError(activityContext, new AppError(TAG, 609, CODE_PROFILE_NOT_FOUND, (String) null)));
- }
- });
- }
-
- /* _____ _ _ _____
- / ____| | | |_ _|
- | | __| | | | | | __ ___ __ __ _ _ __ _ __ ___ _ __ ___
- | | |_ | | | | | | \ \ /\ / / '__/ _` | '_ \| '_ \ / _ \ '__/ __|
- | |__| | |__| |_| |_ \ V V /| | | (_| | |_) | |_) | __/ | \__ \
- \_____|\____/|_____| \_/\_/ |_| \__,_| .__/| .__/ \___|_| |___/
- | | | |
- |_| |*/
- /**
- * Sync all Edziennik data while showing a progress dialog.
- * A wrapper for {@code sync}
- *
- * Does not switch between threads.
- * All callbacks have to be executed on the UI thread.
- *
- * @param app an App singleton instance
- * @param activity a parent activity
- * @param profileId ID of the profile to sync
- * @param dialogTitle a title of the dialog to show
- * @param dialogText dialog's content
- * @param successText a toast to show on success
- */
- public void guiSync(@NonNull App app, @NonNull Activity activity, int profileId, @StringRes int dialogTitle, @StringRes int dialogText, @StringRes int successText) {
- guiSync(app, activity, profileId, dialogTitle, dialogText, successText, (int[])null);
- }
- public void guiSync(@NonNull App app, @NonNull Activity activity, int profileId, @StringRes int dialogTitle, @StringRes int dialogText, @StringRes int successText, int ... featureList) {
- MaterialDialog progressDialog = new MaterialDialog.Builder(activity)
- .title(dialogTitle)
- .content(dialogText)
- .progress(false, PROFILE_MAX_PROGRESS, false)
- .canceledOnTouchOutside(false)
- .show();
- SyncCallback guiSyncCallback = new SyncCallback() {
- @Override
- public void onLoginFirst(List profileList, LoginStore loginStore) {
-
- }
-
- @Override
- public void onSuccess(Context activityContext, ProfileFull profileFull) {
- progressDialog.dismiss();
- Toast.makeText(activityContext, successText, Toast.LENGTH_SHORT).show();
- notifyAndReload();
- // profiles are saved automatically, during app.saveConfig in processFinish
- /*if (activityContext instanceof MainActivity) {
- //((MainActivity) activityContext).reloadCurrentFragment("GuiSync");
- ((MainActivity) activityContext).accountHeaderAddProfiles();
- }*/
- }
-
- @Override
- public void onError(Context activityContext, AppError error) {
- progressDialog.dismiss();
- guiShowErrorDialog((Activity) activityContext, error, R.string.sync_error_dialog_title);
- }
-
- @Override
- public void onProgress(int progressStep) {
- progressDialog.incrementProgress(progressStep);
- }
-
- @Override
- public void onActionStarted(int stringResId) {
- progressDialog.setContent(activity.getString(R.string.sync_action_format, activity.getString(stringResId)));
- }
- };
- app.apiEdziennik.sync(app, activity, guiSyncCallback, profileId, featureList);
- }
- /**
- * Sync all Edziennik data in background.
- * A callback is executed on main thread.
- * A wrapper for {@code sync}
- *
- * @param app an App singleton instance
- * @param activity a parent activity
- * @param profileId ID of the profile to sync
- * @param syncCallback a callback
- * @param feature a feature to sync
- */
- public void guiSyncSilent(@NonNull App app, @NonNull Activity activity, int profileId, SyncCallback syncCallback, int feature) {
- SyncCallback guiSyncCallback = new SyncCallback() {
- @Override
- public void onLoginFirst(List profileList, LoginStore loginStore) {
-
- }
-
- @Override
- public void onSuccess(Context activityContext, ProfileFull profileFull) {
- notifyAndReload();
- syncCallback.onSuccess(activityContext, profileFull);
- }
-
- @Override
- public void onError(Context activityContext, AppError error) {
- syncCallback.onError(activityContext, error);
- }
-
- @Override
- public void onProgress(int progressStep) {
- syncCallback.onProgress(progressStep);
- }
-
- @Override
- public void onActionStarted(int stringResId) {
- syncCallback.onActionStarted(stringResId);
- }
- };
- app.apiEdziennik.sync(app, activity, guiSyncCallback, profileId, feature == FEATURE_ALL ? null : new int[]{feature});
- }
-
- /**
- * Show a dialog allowing the user to choose which features to sync.
- * Handles everything including pre-selecting the features basing on the current fragment.
- *
- * Will execute {@code sync} after the selection is made.
- *
- * A normal progress dialog is shown during the sync.
- *
- * @param app an App singleton instance
- * @param activity a parent activity
- * @param profileId ID of the profile to sync
- * @param dialogTitle a title of the dialog to show
- * @param dialogText dialog's content
- * @param successText a toast to show on success
- * @param currentFeature a feature id representing the currently opened fragment or caller
- */
- public void guiSyncFeature(@NonNull App app,
- @NonNull Activity activity,
- int profileId,
- @StringRes int dialogTitle,
- @StringRes int dialogText,
- @StringRes int successText,
- int currentFeature) {
-
- String[] items = new String[]{
- app.getString(R.string.menu_timetable),
- app.getString(R.string.menu_agenda),
- app.getString(R.string.menu_grades),
- app.getString(R.string.menu_homework),
- app.getString(R.string.menu_notices),
- app.getString(R.string.menu_attendance),
- app.getString(R.string.title_messages_inbox_single),
- app.getString(R.string.title_messages_sent_single),
- app.getString(R.string.menu_announcements)
- };
- int[] itemsIds = new int[]{
- FEATURE_TIMETABLE,
- FEATURE_AGENDA,
- FEATURE_GRADES,
- FEATURE_HOMEWORK,
- FEATURE_NOTICES,
- FEATURE_ATTENDANCE,
- FEATURE_MESSAGES_INBOX,
- FEATURE_MESSAGES_OUTBOX,
- FEATURE_ANNOUNCEMENTS
- };
- int[] selectedIndices;
- if (currentFeature == FEATURE_ALL) {
- selectedIndices = new int[]{0, 1, 2, 3, 4, 5, 6, 7, 8};
- }
- else {
- selectedIndices = new int[]{Arrays.binarySearch(itemsIds, currentFeature)};
- }
-
- MaterialDialog dialog = new MaterialDialog.Builder(activity)
- .title(R.string.sync_feature_title)
- .content(R.string.sync_feature_text)
- .positiveText(R.string.ok)
- .negativeText(R.string.cancel)
- .neutralText(R.string.sync_feature_all)
- .items(items)
- .itemsIds(itemsIds)
- .itemsCallbackMultiChoice(ArrayUtils.toWrapperArray(selectedIndices), (dialog1, which, text) -> {
- dialog1.getActionButton(DialogAction.POSITIVE).setEnabled(which.length > 0);
- return true;
- })
- .alwaysCallMultiChoiceCallback()
- .onPositive(((dialog1, which) -> {
- List featureList = new ArrayList<>();
- for (int i: dialog1.getSelectedIndices()) {
- featureList.add(itemsIds[i]);
- }
- guiSync(app, activity, profileId, dialogTitle, dialogText, successText, ArrayUtils.toPrimitiveArray(featureList));
- }))
- .onNeutral(((dialog1, which) -> {
- guiSync(app, activity, profileId, dialogTitle, dialogText, successText);
- }))
- .show();
-
-
-
- }
-
- public void guiShowArchivedDialog(Activity activity, String profileName) {
- new MaterialDialog.Builder(activity)
- .title(R.string.profile_archived_dialog_title)
- .content(activity.getString(R.string.profile_archived_dialog_text_format, profileName))
- .positiveText(R.string.ok)
- .onPositive(((dialog, which) -> dialog.dismiss()))
- .autoDismiss(false)
- .show();
- }
-
- /* _____ _ _ _____
- / ____| | | |_ _|
- | | __| | | | | | ___ _ __ _ __ ___ _ __ ___
- | | |_ | | | | | | / _ \ '__| '__/ _ \| '__/ __|
- | |__| | |__| |_| |_ | __/ | | | | (_) | | \__ \
- \_____|\____/|_____| \___|_| |_| \___/|_| |__*/
- /**
- * Used for reporting an exception somewhere in the code that is not part of Edziennik APIs.
- *
- * @param activity a parent activity
- * @param errorLine the line of code where the error occurred
- * @param e an Exception object
- */
- public void guiReportException(Activity activity, int errorLine, Exception e) {
- guiReportError(activity, new AppError(TAG, errorLine, CODE_OTHER, "Błąd wewnętrzny aplikacji ("+errorLine+")", null, null, e, null), null);
- }
-
- public void guiShowErrorDialog(Activity activity, @NonNull AppError error, @StringRes int dialogTitle) {
- if (error.errorCode == CODE_PROFILE_ARCHIVED) {
- guiShowArchivedDialog(activity, error.errorText);
- return;
- }
- error.changeIfCodeOther();
- new MaterialDialog.Builder(activity)
- .title(dialogTitle)
- .content(error.asReadableString(activity))
- .positiveText(R.string.ok)
- .onPositive(((dialog, which) -> dialog.dismiss()))
- .neutralText(R.string.sync_error_dialog_report_button)
- .onNeutral(((dialog, which) -> {
- guiReportError(activity, error, dialog);
- }))
- .autoDismiss(false)
- .show();
- }
- public void guiShowErrorSnackbar(MainActivity activity, @NonNull AppError error) {
- if (error.errorCode == CODE_PROFILE_ARCHIVED) {
- guiShowArchivedDialog(activity, error.errorText);
- return;
- }
-
- // TODO: 2019-08-28
- IconicsDrawable icon = new IconicsDrawable(activity)
- .icon(CommunityMaterial.Icon.cmd_alert_circle);
- sizeDp(icon, 20);
- colorInt(icon, Themes.INSTANCE.getPrimaryTextColor(activity));
-
- error.changeIfCodeOther();
- CafeBar.builder(activity)
- .to(activity.findViewById(R.id.coordinator))
- .content(error.asReadableString(activity))
- .icon(icon)
- .positiveText(R.string.more)
- .positiveColor(0xff4caf50)
- .negativeText(R.string.ok)
- .negativeColor(0x66ffffff)
- .onPositive((cafeBar -> guiReportError(activity, error, null)))
- .onNegative((cafeBar -> cafeBar.dismiss()))
- .autoDismiss(false)
- .swipeToDismiss(true)
- .floating(true)
- .show();
- }
- public void guiReportError(Activity activity, AppError error, @Nullable MaterialDialog parentDialogToDisableNeutral) {
- String errorDetails = error.getDetails(activity);
- String htmlErrorDetails = ""+errorDetails+"";
- htmlErrorDetails = htmlErrorDetails.replaceAll(activity.getPackageName(), ""+activity.getPackageName()+"");
- htmlErrorDetails = htmlErrorDetails.replaceAll("\n", "
");
-
- new MaterialDialog.Builder(activity)
- .title(R.string.sync_report_dialog_title)
- .content(Html.fromHtml(htmlErrorDetails))
- .typeface(null, "RobotoMono-Regular.ttf")
- .negativeText(R.string.close)
- .onNegative(((dialog1, which1) -> dialog1.dismiss()))
- .neutralText(R.string.copy_to_clipboard)
- .onNeutral((dialog1, which1) -> {
- ClipboardManager clipboard = (ClipboardManager) activity.getSystemService(CLIPBOARD_SERVICE);
- if (clipboard != null) {
- ClipData clip = ClipData.newPlainText("Error report", errorDetails);
- clipboard.setPrimaryClip(clip);
- Toast.makeText(activity, R.string.copied_to_clipboard, Toast.LENGTH_SHORT).show();
- }
- })
- .autoDismiss(false)
- .positiveText(R.string.sync_report_dialog_button)
- .checkBoxPromptRes(R.string.sync_report_dialog_include_api_response, true, null)
- .onPositive(((dialog1, which1) -> AsyncTask.execute(() -> error.getApiResponse(activity, apiResponse -> {
- new ServerRequest(app, app.requestScheme + APP_URL + "main.php?report", "Edziennik/Report")
- .setBodyParameter("base64_encoded", Base64.encodeToString(errorDetails.getBytes(), Base64.DEFAULT))
- .setBodyParameter("api_response", dialog1.isPromptCheckBoxChecked() ? Base64.encodeToString(apiResponse.getBytes(), Base64.DEFAULT) : "VW5jaGVja2Vk"/*Unchecked*/)
- .run((e, result) -> {
- new Handler(activity.getMainLooper()).post(() -> {
- if (result != null)
- {
- if (result.get("success").getAsBoolean()) {
- Toast.makeText(activity, activity.getString(R.string.crash_report_sent), Toast.LENGTH_SHORT).show();
- dialog1.getActionButton(DialogAction.POSITIVE).setEnabled(false);
- if (parentDialogToDisableNeutral != null)
- parentDialogToDisableNeutral.getActionButton(DialogAction.NEUTRAL).setEnabled(false);
- }
- else {
- Toast.makeText(activity, activity.getString(R.string.crash_report_cannot_send) + ": " + result.get("reason").getAsString(), Toast.LENGTH_LONG).show();
- }
- }
- else
- {
- Toast.makeText(activity, activity.getString(R.string.crash_report_cannot_send)+" brak internetu", Toast.LENGTH_LONG).show();
- }
- });
- });
- }))))
- .show();
- }
-
- /**
- * A method that displays a dialog allowing the user to report an error that has occurred.
- *
- * @param activity a parent activity
- * @param errorCode self-explanatory
- * @param errorText additional error information, that replaces text based on {@code errorCode} if it's {@code CODE_OTHER}
- * @param throwable a {@link Throwable} containing the error details
- * @param apiResponse response of the Edziennik API
- * @param parentDialogToDisableNeutral if not null, an instance of {@link MaterialDialog} in which the neutral button should be disabled after submitting an error report
- */
- public void guiReportError(Activity activity, int errorCode, String errorText, Throwable throwable, String apiResponse, @Nullable MaterialDialog parentDialogToDisableNeutral) {
- // build a string containing the stack trace and the device name + user's registration data
- String contentPlain = "Application Internal Error "+stringErrorType(errorCode)+":\n"+stringErrorCode(activity, errorCode, "")+"\n"+errorText+"\n\n";
- contentPlain += Log.getStackTraceString(throwable);
- String content = ""+contentPlain+"";
- content = content.replaceAll(activity.getPackageName(), ""+activity.getPackageName()+"");
- content = content.replaceAll("\n", "
");
-
- contentPlain += "\n"+Build.MANUFACTURER+"\n"+Build.BRAND+"\n"+Build.MODEL+"\n"+Build.DEVICE+"\n";
- if (app.profile != null && app.profile.getRegistration() == REGISTRATION_ENABLED) {
- contentPlain += "U: "+app.profile.getUsernameId()+"\nS: "+ app.profile.getStudentNameLong() +"\nT: "+app.profile.loginStoreType()+"\n";
- }
- contentPlain += BuildConfig.VERSION_NAME+" "+BuildConfig.BUILD_TYPE+"\nAndroid "+Build.VERSION.RELEASE;
-
- d(TAG, contentPlain);
- d(TAG, apiResponse == null ? "API Response = null" : apiResponse);
-
-
- // show a dialog containing the error details in HTML
- String finalContentPlain = contentPlain;
- new MaterialDialog.Builder(activity)
- .title(R.string.sync_report_dialog_title)
- .content(Html.fromHtml(content))
- .typeface(null, "RobotoMono-Regular.ttf")
- .negativeText(R.string.close)
- .onNegative(((dialog1, which1) -> dialog1.dismiss()))
- .neutralText(R.string.copy_to_clipboard)
- .onNeutral((dialog1, which1) -> {
- ClipboardManager clipboard = (ClipboardManager) activity.getSystemService(CLIPBOARD_SERVICE);
- if (clipboard != null) {
- ClipData clip = ClipData.newPlainText("Error report", finalContentPlain);
- clipboard.setPrimaryClip(clip);
- Toast.makeText(activity, R.string.copied_to_clipboard, Toast.LENGTH_SHORT).show();
- }
- })
- .autoDismiss(false)
- .positiveText(R.string.sync_report_dialog_button)
- .checkBoxPromptRes(R.string.sync_report_dialog_include_api_response, true, null)
- .onPositive(((dialog1, which1) -> {
- // send the error report
- new ServerRequest(app, app.requestScheme + APP_URL + "main.php?report", "Edziennik/Report")
- .setBodyParameter("base64_encoded", Base64.encodeToString(finalContentPlain.getBytes(), Base64.DEFAULT))
- .setBodyParameter("api_response", dialog1.isPromptCheckBoxChecked() ? apiResponse == null ? Base64.encodeToString("NULL XD".getBytes(), Base64.DEFAULT) : Base64.encodeToString(apiResponse.getBytes(), Base64.DEFAULT) : "VW5jaGVja2Vk"/*Unchecked*/)
- .run((e, result) -> {
- new Handler(Looper.getMainLooper()).post(() -> {
- if (result != null)
- {
- if (result.get("success").getAsBoolean()) {
- Toast.makeText(activity, activity.getString(R.string.crash_report_sent), Toast.LENGTH_SHORT).show();
- dialog1.getActionButton(DialogAction.POSITIVE).setEnabled(false);
- if (parentDialogToDisableNeutral != null)
- parentDialogToDisableNeutral.getActionButton(DialogAction.NEUTRAL).setEnabled(false);
- }
- else {
- Toast.makeText(activity, activity.getString(R.string.crash_report_cannot_send) + ": " + result.get("reason").getAsString(), Toast.LENGTH_LONG).show();
- }
- }
- else
- {
- Toast.makeText(activity, activity.getString(R.string.crash_report_cannot_send)+" JsonObject equals null", Toast.LENGTH_LONG).show();
- }
- });
- });
- }))
- .show();
- }
-
- /* _____ __ _ _ _
- | __ \ / _(_) | | |
- | |__) | __ ___ | |_ _| | ___ _ __ ___ _ __ ___ _____ ____ _| |
- | ___/ '__/ _ \| _| | |/ _ \ | '__/ _ \ '_ ` _ \ / _ \ \ / / _` | |
- | | | | | (_) | | | | | __/ | | | __/ | | | | | (_) \ V / (_| | |
- |_| |_| \___/|_| |_|_|\___| |_| \___|_| |_| |_|\___/ \_/ \__,_|*/
- public void guiRemoveProfile(MainActivity activity, int profileId, String profileName) {
- new MaterialDialog.Builder(activity)
- .title(R.string.profile_menu_remove_confirm)
- .content(activity.getString(R.string.profile_menu_remove_confirm_text_format, profileName, profileName))
- .positiveText(R.string.remove)
- .negativeText(R.string.cancel)
- .onPositive(((dialog, which) -> {
- AsyncTask.execute(() -> {
- removeProfile(profileId);
- activity.runOnUiThread(() -> {
- //activity.drawer.loadItem(DRAWER_ITEM_HOME, null, "ProfileRemoving");
- //activity.recreate(DRAWER_ITEM_HOME);
- activity.reloadTarget();
- Toast.makeText(activity, "Profil został usunięty.", Toast.LENGTH_LONG).show();
- });
- });
- }))
- .show();
- }
- public void removeProfile(int profileId) {
- Profile profileObject = app.db.profileDao().getByIdNow(profileId);
- if (profileObject == null)
- return;
- app.db.announcementDao().clear(profileId);
- app.db.attendanceDao().clear(profileId);
- app.db.eventDao().clear(profileId);
- app.db.eventTypeDao().clear(profileId);
- app.db.gradeDao().clear(profileId);
- app.db.gradeCategoryDao().clear(profileId);
- app.db.lessonDao().clear(profileId);
- app.db.lessonChangeDao().clear(profileId);
- app.db.luckyNumberDao().clear(profileId);
- app.db.noticeDao().clear(profileId);
- app.db.subjectDao().clear(profileId);
- app.db.teacherDao().clear(profileId);
- app.db.teamDao().clear(profileId);
- app.db.messageRecipientDao().clear(profileId);
- app.db.messageDao().clear(profileId);
-
- int loginStoreId = profileObject.getLoginStoreId();
- List profilesUsingLoginStore = app.db.profileDao().getIdsByLoginStoreIdNow(loginStoreId);
- if (profilesUsingLoginStore.size() == 1) {
- app.db.loginStoreDao().remove(loginStoreId);
- }
- app.db.profileDao().remove(profileId);
- app.db.metadataDao().deleteAll(profileId);
-
- List toRemove = new ArrayList<>();
- for (Notification notification: app.appConfig.notifications) {
- if (notification.profileId == profileId) {
- toRemove.add(notification);
- }
- }
- app.appConfig.notifications.removeAll(toRemove);
-
- app.profile = null;
- App.profileId = -1;
- }
-}
diff --git a/app/src/main/java/pl/szczodrzynski/edziennik/data/api/EdziennikNotification.kt b/app/src/main/java/pl/szczodrzynski/edziennik/data/api/EdziennikNotification.kt
new file mode 100644
index 00000000..6b2b7533
--- /dev/null
+++ b/app/src/main/java/pl/szczodrzynski/edziennik/data/api/EdziennikNotification.kt
@@ -0,0 +1,144 @@
+/*
+ * Copyright (c) Kuba Szczodrzyński 2019-10-1.
+ */
+
+package pl.szczodrzynski.edziennik.data.api
+
+import android.app.Notification
+import android.app.NotificationManager
+import android.app.PendingIntent
+import android.content.Context
+import androidx.core.app.NotificationCompat
+import androidx.core.app.NotificationCompat.PRIORITY_MIN
+import pl.szczodrzynski.edziennik.App
+import pl.szczodrzynski.edziennik.R
+import pl.szczodrzynski.edziennik.ext.Bundle
+import pl.szczodrzynski.edziennik.ext.pendingIntentFlag
+import pl.szczodrzynski.edziennik.receivers.SzkolnyReceiver
+import kotlin.math.roundToInt
+
+
+class EdziennikNotification(val app: App) {
+ private val notificationManager by lazy { app.getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager }
+
+ private val notificationBuilder: NotificationCompat.Builder by lazy {
+ NotificationCompat.Builder(app, ApiService.NOTIFICATION_API_CHANNEL_ID)
+ .setSmallIcon(R.drawable.ic_notification)
+ .setPriority(PRIORITY_MIN)
+ .setOngoing(true)
+ .setLocalOnly(true)
+ }
+
+ val notification: Notification
+ get() = notificationBuilder.build()
+
+ private var errorCount = 0
+ private var criticalErrorCount = 0
+ var serviceClosed = false
+
+ private fun cancelPendingIntent(taskId: Int): PendingIntent {
+ val intent = SzkolnyReceiver.getIntent(app, Bundle(
+ "task" to "TaskCancelRequest",
+ "taskId" to taskId
+ ))
+ return PendingIntent.getBroadcast(app, 0, intent, PendingIntent.FLAG_UPDATE_CURRENT or pendingIntentFlag()) as PendingIntent
+ }
+ private val closePendingIntent: PendingIntent
+ get() {
+ val intent = SzkolnyReceiver.getIntent(app, Bundle(
+ "task" to "ServiceCloseRequest"
+ ))
+ return PendingIntent.getBroadcast(app, 0, intent, pendingIntentFlag()) as PendingIntent
+ }
+
+ private fun errorCountText(): String? {
+ var result = ""
+ if (criticalErrorCount > 0) {
+ result += app.resources.getQuantityString(R.plurals.critical_errors_format, criticalErrorCount, criticalErrorCount)
+ }
+ if (criticalErrorCount > 0 && errorCount > 0) {
+ result += ", "
+ }
+ if (errorCount > 0) {
+ result += app.resources.getQuantityString(R.plurals.normal_errors_format, errorCount, errorCount)
+ }
+ return if (result.isEmpty()) null else result
+ }
+
+ fun setIdle(): EdziennikNotification {
+ notificationBuilder.setContentTitle(app.getString(R.string.edziennik_notification_api_title))
+ notificationBuilder.setProgress(0, 0, false)
+ notificationBuilder.apply {
+ val str = app.getString(R.string.edziennik_notification_api_text)
+ setStyle(NotificationCompat.BigTextStyle().bigText(str))
+ setContentText(str)
+ }
+ setCloseAction()
+ return this
+ }
+
+ fun addError(): EdziennikNotification {
+ errorCount++
+ return this
+ }
+ fun setCriticalError(): EdziennikNotification {
+ criticalErrorCount++
+ notificationBuilder.setContentTitle(app.getString(R.string.edziennik_notification_api_error_title))
+ notificationBuilder.setProgress(0, 0, false)
+ notificationBuilder.apply {
+ val str = errorCountText()
+ setStyle(NotificationCompat.BigTextStyle().bigText(str))
+ setContentText(str)
+ }
+ setCloseAction()
+ return this
+ }
+
+ fun setProgress(progress: Float): EdziennikNotification {
+ notificationBuilder.setProgress(100, progress.roundToInt(), progress < 0f)
+ return this
+ }
+ fun setProgressText(progressText: String?): EdziennikNotification {
+ notificationBuilder.setContentTitle(progressText)
+ return this
+ }
+
+ fun setCurrentTask(taskId: Int, progressText: String?): EdziennikNotification {
+ notificationBuilder.setProgress(100, 0, true)
+ notificationBuilder.setContentTitle(progressText)
+ notificationBuilder.apply {
+ val str = errorCountText()
+ setStyle(NotificationCompat.BigTextStyle().bigText(str))
+ setContentText(str)
+ }
+ setCancelAction(taskId)
+ return this
+ }
+
+ fun setCloseAction(): EdziennikNotification {
+ notificationBuilder.mActions.clear()
+ notificationBuilder.addAction(
+ NotificationCompat.Action(
+ R.drawable.ic_notification,
+ app.getString(R.string.edziennik_notification_api_close),
+ closePendingIntent
+ ))
+ return this
+ }
+ private fun setCancelAction(taskId: Int) {
+ notificationBuilder.mActions.clear()
+ notificationBuilder.addAction(
+ NotificationCompat.Action(
+ R.drawable.ic_notification,
+ app.getString(R.string.edziennik_notification_api_cancel),
+ cancelPendingIntent(taskId)
+ ))
+ }
+
+ fun post() {
+ if (serviceClosed)
+ return
+ notificationManager.notify(app.notificationChannelsManager.sync.id, notification)
+ }
+
+}
diff --git a/app/src/main/java/pl/szczodrzynski/edziennik/data/api/EndpointChooser.kt b/app/src/main/java/pl/szczodrzynski/edziennik/data/api/EndpointChooser.kt
new file mode 100644
index 00000000..de834344
--- /dev/null
+++ b/app/src/main/java/pl/szczodrzynski/edziennik/data/api/EndpointChooser.kt
@@ -0,0 +1,118 @@
+package pl.szczodrzynski.edziennik.data.api
+
+import pl.szczodrzynski.edziennik.data.api.models.Data
+import pl.szczodrzynski.edziennik.data.api.models.Feature
+import pl.szczodrzynski.edziennik.data.db.entity.EndpointTimer
+import pl.szczodrzynski.edziennik.data.db.entity.SYNC_ALWAYS
+import pl.szczodrzynski.edziennik.data.db.entity.SYNC_NEVER
+import pl.szczodrzynski.edziennik.data.db.enums.FeatureType
+import pl.szczodrzynski.edziennik.data.db.enums.LoginMethod
+import pl.szczodrzynski.edziennik.ext.getFeatureTypesNecessary
+import pl.szczodrzynski.edziennik.ext.getFeatureTypesUnnecessary
+import pl.szczodrzynski.edziennik.ext.isNotNullNorEmpty
+
+fun Data.prepare(
+ features: List,
+ featureTypes: Set?,
+ onlyEndpoints: Set?,
+) {
+ val loginType = this.loginStore.type
+ val possibleLoginMethods = this.loginMethods.toMutableList()
+ possibleLoginMethods += LoginMethod.values().filter {
+ it.loginType == loginType && it.isPossible?.invoke(profile, loginStore) != false
+ }
+
+ //var highestLoginMethod = 0
+ var possibleFeatures = mutableListOf()
+ val requiredLoginMethods = mutableListOf()
+
+ val syncFeatureTypes = when {
+ featureTypes.isNotNullNorEmpty() -> featureTypes!!
+ else -> getFeatureTypesUnnecessary()
+ } + getFeatureTypesNecessary()
+ val forceFeatureType = featureTypes?.singleOrNull()
+
+ this.targetEndpoints.clear()
+ this.targetLoginMethods.clear()
+
+ // get all endpoints for every feature, only if possible to login and possible/necessary to sync
+ for (featureId in syncFeatureTypes) {
+ possibleFeatures += features.filter {
+ it.featureType == featureId // feature ID matches
+ && possibleLoginMethods.containsAll(it.requiredLoginMethods) // is possible to login
+ && it.shouldSync?.invoke(this) ?: true // is necessary/possible to sync
+ }
+ }
+
+ val timestamp = System.currentTimeMillis()
+
+ possibleFeatures = possibleFeatures
+ // sort the endpoint list by feature ID and priority
+ .sortedWith(compareBy(Feature::featureType, Feature::priority))
+ // select only the most important endpoint for each feature
+ .distinctBy { it.featureType }
+ .toMutableList()
+
+ for (feature in possibleFeatures) {
+ // add all endpoint IDs and required login methods, filtering using timers
+ feature.endpoints.forEach { endpoint ->
+ if (onlyEndpoints?.contains(endpoint.first) == false)
+ return@forEach
+ val timer = this.endpointTimers
+ .singleOrNull { it.endpointId == endpoint.first }
+ ?: EndpointTimer(this.profileId, endpoint.first)
+ if (
+ onlyEndpoints?.contains(endpoint.first) == true ||
+ timer.nextSync == SYNC_ALWAYS ||
+ forceFeatureType != null && timer.featureType == forceFeatureType ||
+ timer.nextSync != SYNC_NEVER && timer.nextSync < timestamp
+ ) {
+ this.targetEndpoints[endpoint.first] = timer.lastSync
+ requiredLoginMethods += endpoint.second
+ }
+ }
+ }
+
+ // check every login method for any dependencies
+ for (loginMethod in requiredLoginMethods) {
+ var requiredLoginMethod: LoginMethod? = loginMethod
+ while (requiredLoginMethod != null) {
+ this.targetLoginMethods += requiredLoginMethod
+ requiredLoginMethod = requiredLoginMethod.requiredLoginMethod?.invoke(this.profile, this.loginStore)
+ }
+ }
+
+ // sort and distinct every login method and endpoint
+ this.targetLoginMethods = this.targetLoginMethods.toHashSet().toMutableList()
+ this.targetLoginMethods.sort()
+
+ //data.targetEndpointIds = data.targetEndpointIds.toHashSet().toMutableList()
+ //data.targetEndpointIds.sort()
+
+ progressCount = targetLoginMethods.size + targetEndpoints.size
+ progressStep = if (progressCount <= 0) 0f else 100f / progressCount.toFloat()
+}
+
+fun Data.prepareFor(loginMethod: LoginMethod) {
+ val loginType = loginStore.type
+ val possibleLoginMethods = this.loginMethods.toMutableList()
+ possibleLoginMethods += LoginMethod.values().filter {
+ it.loginType == loginType && it.isPossible?.invoke(profile, loginStore) != false
+ }
+
+ this.targetLoginMethods.clear()
+
+ // check the login method for any dependencies
+ var requiredLoginMethod: LoginMethod? = loginMethod
+ while (requiredLoginMethod != null) {
+ this.targetLoginMethods += requiredLoginMethod
+ requiredLoginMethod = requiredLoginMethod.requiredLoginMethod?.invoke(this.profile, this.loginStore)
+ }
+
+ // sort and distinct every login method
+ this.targetLoginMethods = this.targetLoginMethods.toHashSet().toMutableList()
+ this.targetLoginMethods.sort()
+
+ progressCount = 0
+ progressStep = 0f
+}
diff --git a/app/src/main/java/pl/szczodrzynski/edziennik/data/api/Errors.kt b/app/src/main/java/pl/szczodrzynski/edziennik/data/api/Errors.kt
new file mode 100644
index 00000000..93a7228f
--- /dev/null
+++ b/app/src/main/java/pl/szczodrzynski/edziennik/data/api/Errors.kt
@@ -0,0 +1,228 @@
+/*
+ * Copyright (c) Kuba Szczodrzyński 2019-9-21.
+ */
+
+package pl.szczodrzynski.edziennik.data.api
+
+/*const val CODE_OTHER = 0
+const val CODE_OK = 1
+const val CODE_NO_INTERNET = 10
+const val CODE_SSL_ERROR = 13
+const val CODE_ARCHIVED = 5
+const val CODE_MAINTENANCE = 6
+const val CODE_LOGIN_ERROR = 7
+const val CODE_ACCOUNT_MISMATCH = 8
+const val CODE_APP_SERVER_ERROR = 9
+const val CODE_MULTIACCOUNT_SETUP = 12
+const val CODE_TIMEOUT = 11
+const val CODE_PROFILE_NOT_FOUND = 14
+const val CODE_ATTACHMENT_NOT_AVAILABLE = 28
+const val CODE_INVALID_LOGIN = 2
+const val CODE_INVALID_SERVER_ADDRESS = 21
+const val CODE_INVALID_SCHOOL_NAME = 22
+const val CODE_INVALID_DEVICE = 23
+const val CODE_OLD_PASSWORD = 4
+const val CODE_INVALID_TOKEN = 24
+const val CODE_EXPIRED_TOKEN = 27
+const val CODE_INVALID_SYMBOL = 25
+const val CODE_INVALID_PIN = 26
+const val CODE_LIBRUS_NOT_ACTIVATED = 29
+const val CODE_SYNERGIA_NOT_ACTIVATED = 32
+const val CODE_LIBRUS_DISCONNECTED = 31
+const val CODE_PROFILE_ARCHIVED = 30*/
+
+const val ERROR_APP_CRASH = 1
+const val ERROR_EXCEPTION = 2
+const val ERROR_API_EXCEPTION = 3
+const val ERROR_MESSAGE_NOT_SENT = 10
+
+const val ERROR_REQUEST_FAILURE = 50
+const val ERROR_REQUEST_HTTP_400 = 51
+const val ERROR_REQUEST_HTTP_401 = 52
+const val ERROR_REQUEST_HTTP_403 = 53
+const val ERROR_REQUEST_HTTP_404 = 54
+const val ERROR_REQUEST_HTTP_405 = 55
+const val ERROR_REQUEST_HTTP_410 = 56
+const val ERROR_REQUEST_HTTP_424 = 57
+const val ERROR_REQUEST_HTTP_500 = 58
+const val ERROR_REQUEST_HTTP_503 = 59
+const val ERROR_REQUEST_FAILURE_HOSTNAME_NOT_FOUND = 60
+const val ERROR_REQUEST_FAILURE_TIMEOUT = 61
+const val ERROR_REQUEST_FAILURE_NO_INTERNET = 62
+const val ERROR_REQUEST_FAILURE_SSL_ERROR = 63
+const val ERROR_RESPONSE_EMPTY = 100
+const val ERROR_LOGIN_DATA_MISSING = 101
+const val ERROR_PROFILE_MISSING = 105
+const val ERROR_PROFILE_ARCHIVED = 106
+const val ERROR_INVALID_LOGIN_MODE = 110
+const val ERROR_LOGIN_METHOD_NOT_SATISFIED = 111
+const val ERROR_NOT_IMPLEMENTED = 112
+const val ERROR_FILE_DOWNLOAD = 113
+const val ERROR_REQUIRES_USER_ACTION = 114
+
+const val ERROR_API_PDO_ERROR = 5000
+const val ERROR_API_INVALID_CLIENT = 5001
+const val ERROR_API_INVALID_ARGUMENT = 5002
+const val ERROR_API_INVALID_SIGNATURE = 5003
+const val ERROR_API_MISSING_SCOPES = 5004
+const val ERROR_API_RESOURCE_NOT_FOUND = 5005
+const val ERROR_API_INTERNAL_SERVER_ERROR = 5006
+const val ERROR_API_PHP_E_ERROR = 5007
+const val ERROR_API_PHP_E_WARNING = 5008
+const val ERROR_API_PHP_E_PARSE = 5009
+const val ERROR_API_PHP_E_NOTICE = 5010
+const val ERROR_API_PHP_E_OTHER = 5011
+const val ERROR_API_MAINTENANCE = 5012
+const val ERROR_API_MISSING_ARGUMENT = 5013
+const val ERROR_API_PAYLOAD_EMPTY = 5014
+const val ERROR_API_INVALID_ACTION = 5015
+const val ERROR_API_UPDATE_NOT_FOUND = 5016
+const val ERROR_API_INVALID_DEVICEID_USERCODE = 5017
+const val ERROR_API_INVALID_PAIRTOKEN = 5018
+const val ERROR_API_INVALID_BROWSERID = 5019
+const val ERROR_API_INVALID_DEVICEID = 5020
+const val ERROR_API_INVALID_DEVICEID_BROWSERID = 5021
+const val ERROR_API_HELP_CATEGORY_NOT_FOUND = 5022
+
+const val CODE_INTERNAL_LIBRUS_ACCOUNT_410 = 120
+const val CODE_INTERNAL_LIBRUS_SYNERGIA_EXPIRED = 121
+const val ERROR_LOGIN_LIBRUS_API_CAPTCHA_NEEDED = 124
+const val ERROR_LOGIN_LIBRUS_API_CONNECTION_PROBLEMS = 125
+const val ERROR_LOGIN_LIBRUS_API_INVALID_CLIENT = 126
+const val ERROR_LOGIN_LIBRUS_API_REG_ACCEPT_NEEDED = 127
+const val ERROR_LOGIN_LIBRUS_API_CHANGE_PASSWORD_ERROR = 128
+const val ERROR_LOGIN_LIBRUS_API_PASSWORD_CHANGE_REQUIRED = 129
+const val ERROR_LOGIN_LIBRUS_API_INVALID_LOGIN = 130
+const val ERROR_LOGIN_LIBRUS_API_OTHER = 131
+const val ERROR_LOGIN_LIBRUS_PORTAL_CSRF_MISSING = 132
+const val ERROR_LOGIN_LIBRUS_PORTAL_NOT_ACTIVATED = 133
+const val ERROR_LOGIN_LIBRUS_PORTAL_ACTION_ERROR = 134
+const val ERROR_LOGIN_LIBRUS_PORTAL_SYNERGIA_TOKEN_MISSING = 139
+const val ERROR_LIBRUS_API_TOKEN_EXPIRED = 140
+const val ERROR_LIBRUS_API_INSUFFICIENT_SCOPES = 141
+const val ERROR_LIBRUS_API_OTHER = 142
+const val ERROR_LIBRUS_API_ACCESS_DENIED = 143
+const val ERROR_LIBRUS_API_RESOURCE_NOT_FOUND = 144
+const val ERROR_LIBRUS_API_DATA_NOT_FOUND = 145
+const val ERROR_LIBRUS_API_TIMETABLE_NOT_PUBLIC = 146
+const val ERROR_LIBRUS_API_RESOURCE_ACCESS_DENIED = 147
+const val ERROR_LIBRUS_API_INVALID_REQUEST_PARAMS = 148
+const val ERROR_LIBRUS_API_INCORRECT_ENDPOINT = 149
+const val ERROR_LIBRUS_API_LUCKY_NUMBER_NOT_ACTIVE = 150
+const val ERROR_LIBRUS_API_NOTES_NOT_ACTIVE = 151
+const val ERROR_LOGIN_LIBRUS_SYNERGIA_NO_TOKEN = 152
+const val ERROR_LOGIN_LIBRUS_SYNERGIA_TOKEN_INVALID = 153
+const val ERROR_LOGIN_LIBRUS_SYNERGIA_NO_SESSION_ID = 154
+const val ERROR_LIBRUS_MESSAGES_ACCESS_DENIED = 155
+const val ERROR_LIBRUS_SYNERGIA_ACCESS_DENIED = 156
+const val ERROR_LOGIN_LIBRUS_MESSAGES_NO_SESSION_ID = 157
+const val ERROR_LIBRUS_PORTAL_ACCESS_DENIED = 158
+const val ERROR_LIBRUS_PORTAL_API_DISABLED = 159
+const val ERROR_LIBRUS_PORTAL_SYNERGIA_DISCONNECTED = 160
+const val ERROR_LIBRUS_PORTAL_OTHER = 161
+const val ERROR_LIBRUS_PORTAL_SYNERGIA_NOT_FOUND = 162
+const val ERROR_LOGIN_LIBRUS_PORTAL_OTHER = 163
+const val ERROR_LOGIN_LIBRUS_PORTAL_CODE_EXPIRED = 164
+const val ERROR_LOGIN_LIBRUS_PORTAL_CODE_REVOKED = 165
+const val ERROR_LOGIN_LIBRUS_PORTAL_NO_CLIENT_ID = 166
+const val ERROR_LOGIN_LIBRUS_PORTAL_NO_CODE = 167
+const val ERROR_LOGIN_LIBRUS_PORTAL_NO_REFRESH = 168
+const val ERROR_LOGIN_LIBRUS_PORTAL_NO_REDIRECT = 169
+const val ERROR_LOGIN_LIBRUS_PORTAL_UNSUPPORTED_GRANT = 170
+const val ERROR_LOGIN_LIBRUS_PORTAL_INVALID_CLIENT_ID = 171
+const val ERROR_LOGIN_LIBRUS_PORTAL_REFRESH_INVALID = 172
+const val ERROR_LOGIN_LIBRUS_PORTAL_REFRESH_REVOKED = 173
+const val ERROR_LIBRUS_SYNERGIA_OTHER = 174
+const val ERROR_LIBRUS_SYNERGIA_MAINTENANCE = 175
+const val ERROR_LIBRUS_MESSAGES_MAINTENANCE = 176
+const val ERROR_LIBRUS_MESSAGES_ERROR = 177
+const val ERROR_LIBRUS_MESSAGES_OTHER = 178
+const val ERROR_LOGIN_LIBRUS_MESSAGES_INVALID_LOGIN = 179
+const val ERROR_LOGIN_LIBRUS_PORTAL_INVALID_LOGIN = 180
+const val ERROR_LIBRUS_API_MAINTENANCE = 181
+const val ERROR_LIBRUS_PORTAL_MAINTENANCE = 182
+const val ERROR_LIBRUS_API_NOTICEBOARD_PROBLEM = 183
+const val ERROR_LOGIN_LIBRUS_PORTAL_CSRF_EXPIRED = 184
+const val ERROR_LIBRUS_API_DEVICE_REGISTERED = 185
+const val ERROR_LIBRUS_MESSAGES_NOT_FOUND = 186
+const val ERROR_LOGIN_LIBRUS_API_INVALID_REQUEST = 187
+const val ERROR_LIBRUS_MESSAGES_ATTACHMENT_NOT_FOUND = 188
+const val ERROR_LOGIN_LIBRUS_MESSAGES_TIMEOUT = 189
+
+const val ERROR_LOGIN_MOBIDZIENNIK_WEB_INVALID_LOGIN = 201
+const val ERROR_LOGIN_MOBIDZIENNIK_WEB_OLD_PASSWORD = 202
+const val ERROR_LOGIN_MOBIDZIENNIK_WEB_INVALID_DEVICE = 203
+const val ERROR_LOGIN_MOBIDZIENNIK_WEB_ARCHIVED = 204
+const val ERROR_LOGIN_MOBIDZIENNIK_WEB_MAINTENANCE = 205
+const val ERROR_LOGIN_MOBIDZIENNIK_WEB_INVALID_ADDRESS = 206
+const val ERROR_LOGIN_MOBIDZIENNIK_WEB_OTHER = 210
+const val ERROR_MOBIDZIENNIK_WEB_ACCESS_DENIED = 211
+const val ERROR_MOBIDZIENNIK_WEB_NO_SESSION_KEY = 212
+const val ERROR_MOBIDZIENNIK_WEB_NO_SESSION_VALUE = 216
+const val ERROR_MOBIDZIENNIK_WEB_NO_SERVER_ID = 213
+const val ERROR_MOBIDZIENNIK_WEB_INVALID_RESPONSE = 214
+const val ERROR_LOGIN_MOBIDZIENNIK_WEB_NO_SESSION_ID = 215
+const val ERROR_LOGIN_MOBIDZIENNIK_API2_INVALID_LOGIN = 216
+const val ERROR_LOGIN_MOBIDZIENNIK_API2_OTHER = 217
+const val ERROR_MOBIDZIENNIK_WEB_SERVER_PROBLEM = 218
+
+const val ERROR_LOGIN_VULCAN_INVALID_SYMBOL = 301
+const val ERROR_LOGIN_VULCAN_INVALID_TOKEN = 302
+const val ERROR_LOGIN_VULCAN_INVALID_PIN_0_REMAINING = 310
+const val ERROR_LOGIN_VULCAN_INVALID_PIN_1_REMAINING = 311
+const val ERROR_LOGIN_VULCAN_INVALID_PIN_2_REMAINING = 312
+const val ERROR_LOGIN_VULCAN_EXPIRED_TOKEN = 321
+const val ERROR_LOGIN_VULCAN_NO_PUPILS = 331
+const val ERROR_VULCAN_ATTACHMENT_DOWNLOAD = 343
+const val ERROR_VULCAN_WEB_DATA_MISSING = 344
+const val ERROR_VULCAN_WEB_429 = 345
+const val ERROR_VULCAN_WEB_OTHER = 346
+const val ERROR_VULCAN_WEB_NO_CERTIFICATE = 347
+const val ERROR_VULCAN_WEB_NO_REGISTER = 348
+const val ERROR_VULCAN_WEB_CERTIFICATE_EXPIRED = 349
+const val ERROR_VULCAN_WEB_LOGGED_OUT = 350
+const val ERROR_VULCAN_WEB_CERTIFICATE_POST_FAILED = 351
+const val ERROR_VULCAN_WEB_GRADUATE_ACCOUNT = 352
+const val ERROR_VULCAN_WEB_NO_SCHOOLS = 353
+const val ERROR_VULCAN_HEBE_OTHER = 354
+const val ERROR_VULCAN_HEBE_SIGNATURE_ERROR = 360
+const val ERROR_VULCAN_HEBE_INVALID_PAYLOAD = 361
+const val ERROR_VULCAN_HEBE_FIREBASE_ERROR = 362
+const val ERROR_VULCAN_HEBE_CERTIFICATE_GONE = 363
+const val ERROR_VULCAN_HEBE_SERVER_ERROR = 364
+const val ERROR_VULCAN_HEBE_ENTITY_NOT_FOUND = 365
+const val ERROR_VULCAN_HEBE_MISSING_SENDER_ENTRY = 366
+const val ERROR_VULCAN_API_DEPRECATED = 390
+
+const val ERROR_LOGIN_PODLASIE_API_INVALID_TOKEN = 601
+const val ERROR_LOGIN_PODLASIE_API_DEVICE_LIMIT = 602
+const val ERROR_PODLASIE_API_NO_TOKEN = 630
+const val ERROR_PODLASIE_API_OTHER = 631
+const val ERROR_PODLASIE_API_DATA_MISSING = 632
+
+const val ERROR_USOS_OAUTH_GOT_DIFFERENT_TOKEN = 702
+const val ERROR_USOS_OAUTH_INCOMPLETE_RESPONSE = 703
+const val ERROR_USOS_NO_STUDENT_PROGRAMMES = 704
+const val ERROR_USOS_API_INCOMPLETE_RESPONSE = 705
+const val ERROR_USOS_API_MISSING_RESPONSE = 706
+
+const val ERROR_TEMPLATE_WEB_OTHER = 801
+
+const val EXCEPTION_API_TASK = 900
+const val EXCEPTION_LOGIN_LIBRUS_API_TOKEN = 901
+const val EXCEPTION_LOGIN_LIBRUS_PORTAL_TOKEN = 902
+const val EXCEPTION_LIBRUS_PORTAL_SYNERGIA_TOKEN = 903
+const val EXCEPTION_LIBRUS_API_REQUEST = 904
+const val EXCEPTION_LIBRUS_SYNERGIA_REQUEST = 905
+const val EXCEPTION_MOBIDZIENNIK_WEB_REQUEST = 906
+const val EXCEPTION_MOBIDZIENNIK_WEB_FILE_REQUEST = 908
+const val EXCEPTION_LIBRUS_MESSAGES_FILE_REQUEST = 909
+const val EXCEPTION_NOTIFY = 910
+const val EXCEPTION_LIBRUS_MESSAGES_REQUEST = 911
+const val ERROR_ONEDRIVE_DOWNLOAD = 930
+const val EXCEPTION_VULCAN_WEB_LOGIN = 931
+const val EXCEPTION_VULCAN_WEB_REQUEST = 932
+const val EXCEPTION_PODLASIE_API_REQUEST = 940
+const val EXCEPTION_VULCAN_HEBE_REQUEST = 950
+
+const val LOGIN_NO_ARGUMENTS = 1201
diff --git a/app/src/main/java/pl/szczodrzynski/edziennik/data/api/Iuczniowie.java b/app/src/main/java/pl/szczodrzynski/edziennik/data/api/Iuczniowie.java
deleted file mode 100644
index 33c69671..00000000
--- a/app/src/main/java/pl/szczodrzynski/edziennik/data/api/Iuczniowie.java
+++ /dev/null
@@ -1,1710 +0,0 @@
-package pl.szczodrzynski.edziennik.data.api;
-
-import android.content.Context;
-import android.graphics.Color;
-import android.os.AsyncTask;
-import android.os.Handler;
-import android.util.Log;
-
-import androidx.annotation.NonNull;
-import androidx.annotation.Nullable;
-import androidx.core.util.Pair;
-
-import com.crashlytics.android.Crashlytics;
-import com.google.gson.JsonArray;
-import com.google.gson.JsonElement;
-import com.google.gson.JsonNull;
-import com.google.gson.JsonObject;
-
-import java.util.ArrayList;
-import java.util.Collections;
-import java.util.List;
-import java.util.Map;
-import java.util.regex.Matcher;
-import java.util.regex.Pattern;
-
-import im.wangchao.mhttp.Request;
-import im.wangchao.mhttp.Response;
-import im.wangchao.mhttp.body.MediaTypeUtils;
-import im.wangchao.mhttp.callback.JsonArrayCallbackHandler;
-import im.wangchao.mhttp.callback.JsonCallbackHandler;
-import im.wangchao.mhttp.callback.TextCallbackHandler;
-import okhttp3.Cookie;
-import okhttp3.HttpUrl;
-import pl.szczodrzynski.edziennik.App;
-import pl.szczodrzynski.edziennik.BuildConfig;
-import pl.szczodrzynski.edziennik.R;
-import pl.szczodrzynski.edziennik.data.api.interfaces.AttachmentGetCallback;
-import pl.szczodrzynski.edziennik.data.api.interfaces.EdziennikInterface;
-import pl.szczodrzynski.edziennik.data.api.interfaces.LoginCallback;
-import pl.szczodrzynski.edziennik.data.api.interfaces.MessageGetCallback;
-import pl.szczodrzynski.edziennik.data.api.interfaces.RecipientListGetCallback;
-import pl.szczodrzynski.edziennik.data.api.interfaces.SyncCallback;
-import pl.szczodrzynski.edziennik.data.db.modules.announcements.Announcement;
-import pl.szczodrzynski.edziennik.data.db.modules.attendance.Attendance;
-import pl.szczodrzynski.edziennik.data.db.modules.events.Event;
-import pl.szczodrzynski.edziennik.data.db.modules.grades.Grade;
-import pl.szczodrzynski.edziennik.data.db.modules.lessons.Lesson;
-import pl.szczodrzynski.edziennik.data.db.modules.lessons.LessonChange;
-import pl.szczodrzynski.edziennik.data.db.modules.login.LoginStore;
-import pl.szczodrzynski.edziennik.data.db.modules.luckynumber.LuckyNumber;
-import pl.szczodrzynski.edziennik.data.db.modules.messages.Message;
-import pl.szczodrzynski.edziennik.data.db.modules.messages.MessageFull;
-import pl.szczodrzynski.edziennik.data.db.modules.messages.MessageRecipient;
-import pl.szczodrzynski.edziennik.data.db.modules.messages.MessageRecipientFull;
-import pl.szczodrzynski.edziennik.data.db.modules.metadata.Metadata;
-import pl.szczodrzynski.edziennik.data.db.modules.notices.Notice;
-import pl.szczodrzynski.edziennik.data.db.modules.profiles.Profile;
-import pl.szczodrzynski.edziennik.data.db.modules.profiles.ProfileFull;
-import pl.szczodrzynski.edziennik.data.db.modules.subjects.Subject;
-import pl.szczodrzynski.edziennik.data.db.modules.teachers.Teacher;
-import pl.szczodrzynski.edziennik.data.db.modules.teams.Team;
-import pl.szczodrzynski.edziennik.ui.modules.messages.MessagesComposeInfo;
-import pl.szczodrzynski.edziennik.utils.models.Date;
-import pl.szczodrzynski.edziennik.utils.models.Endpoint;
-import pl.szczodrzynski.edziennik.utils.models.Time;
-import pl.szczodrzynski.edziennik.utils.models.Week;
-
-import static pl.szczodrzynski.edziennik.data.api.AppError.CODE_INVALID_LOGIN;
-import static pl.szczodrzynski.edziennik.data.api.AppError.CODE_INVALID_SCHOOL_NAME;
-import static pl.szczodrzynski.edziennik.data.api.AppError.CODE_MAINTENANCE;
-import static pl.szczodrzynski.edziennik.data.api.AppError.CODE_OTHER;
-import static pl.szczodrzynski.edziennik.data.db.modules.attendance.Attendance.TYPE_ABSENT;
-import static pl.szczodrzynski.edziennik.data.db.modules.attendance.Attendance.TYPE_ABSENT_EXCUSED;
-import static pl.szczodrzynski.edziennik.data.db.modules.attendance.Attendance.TYPE_BELATED;
-import static pl.szczodrzynski.edziennik.data.db.modules.attendance.Attendance.TYPE_PRESENT;
-import static pl.szczodrzynski.edziennik.data.db.modules.attendance.Attendance.TYPE_RELEASED;
-import static pl.szczodrzynski.edziennik.data.db.modules.grades.Grade.TYPE_SEMESTER1_FINAL;
-import static pl.szczodrzynski.edziennik.data.db.modules.grades.Grade.TYPE_SEMESTER1_PROPOSED;
-import static pl.szczodrzynski.edziennik.data.db.modules.grades.Grade.TYPE_YEAR_FINAL;
-import static pl.szczodrzynski.edziennik.data.db.modules.grades.Grade.TYPE_YEAR_PROPOSED;
-import static pl.szczodrzynski.edziennik.data.db.modules.lessons.LessonChange.TYPE_CANCELLED;
-import static pl.szczodrzynski.edziennik.data.db.modules.lessons.LessonChange.TYPE_CHANGE;
-import static pl.szczodrzynski.edziennik.data.db.modules.messages.Message.TYPE_DELETED;
-import static pl.szczodrzynski.edziennik.data.db.modules.messages.Message.TYPE_RECEIVED;
-import static pl.szczodrzynski.edziennik.data.db.modules.messages.Message.TYPE_SENT;
-import static pl.szczodrzynski.edziennik.data.db.modules.notices.Notice.TYPE_NEGATIVE;
-import static pl.szczodrzynski.edziennik.data.db.modules.notices.Notice.TYPE_NEUTRAL;
-import static pl.szczodrzynski.edziennik.data.db.modules.notices.Notice.TYPE_POSITIVE;
-import static pl.szczodrzynski.edziennik.utils.Utils.crc16;
-import static pl.szczodrzynski.edziennik.utils.Utils.crc32;
-import static pl.szczodrzynski.edziennik.utils.Utils.d;
-import static pl.szczodrzynski.edziennik.utils.Utils.getWordGradeValue;
-
-public class Iuczniowie implements EdziennikInterface {
- public Iuczniowie(App app) {
- this.app = app;
- }
-
- private static final String TAG = "api.Iuczniowie";
- private static String IDZIENNIK_URL = "https://iuczniowie.progman.pl/idziennik";
- private static final String userAgent = "Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/70.0.3538.102 Safari/537.36";
-
- private App app;
- private Context activityContext = null;
- private SyncCallback callback = null;
- private int profileId = -1;
- private Profile profile = null;
- private LoginStore loginStore = null;
- private boolean fullSync = true;
- private Date today = Date.getToday();
- private List targetEndpoints = new ArrayList<>();
-
- // PROGRESS
- private static final int PROGRESS_LOGIN = 10;
- private int PROGRESS_COUNT = 1;
- private int PROGRESS_STEP = (90/PROGRESS_COUNT);
-
- private int onlyFeature = FEATURE_ALL;
-
- private List teamList;
- private List teacherList;
- private List subjectList;
- private List lessonList;
- private List lessonChangeList;
- private List gradeList;
- private List eventList;
- private List noticeList;
- private List attendanceList;
- private List