2 * Copyright © 2021 Damyan Ivanov.
3 * This file is part of MoLe.
4 * MoLe is free software: you can distribute it and/or modify it
5 * under the term of the GNU General Public License as published by
6 * the Free Software Foundation, either version 3 of the License, or
7 * (at your opinion), any later version.
9 * MoLe is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License terms for details.
14 * You should have received a copy of the GNU General Public License
15 * along with MoLe. If not, see <https://www.gnu.org/licenses/>.
18 package net.ktnx.mobileledger.ui.new_transaction;
20 import android.content.Context;
21 import android.content.Intent;
22 import android.database.AbstractCursor;
23 import android.database.Cursor;
24 import android.os.Bundle;
25 import android.os.ParcelFormatException;
26 import android.text.TextUtils;
27 import android.util.TypedValue;
28 import android.view.Menu;
29 import android.view.MenuItem;
30 import android.view.View;
32 import androidx.activity.result.ActivityResultLauncher;
33 import androidx.annotation.NonNull;
34 import androidx.core.view.MenuCompat;
35 import androidx.lifecycle.LiveData;
36 import androidx.lifecycle.ViewModelProvider;
37 import androidx.navigation.NavController;
38 import androidx.navigation.fragment.NavHostFragment;
40 import com.google.android.material.dialog.MaterialAlertDialogBuilder;
41 import com.google.android.material.snackbar.Snackbar;
43 import net.ktnx.mobileledger.BuildConfig;
44 import net.ktnx.mobileledger.R;
45 import net.ktnx.mobileledger.async.AsyncCrasher;
46 import net.ktnx.mobileledger.async.DescriptionSelectedCallback;
47 import net.ktnx.mobileledger.async.SendTransactionTask;
48 import net.ktnx.mobileledger.async.TaskCallback;
49 import net.ktnx.mobileledger.databinding.ActivityNewTransactionBinding;
50 import net.ktnx.mobileledger.db.DB;
51 import net.ktnx.mobileledger.db.TemplateHeader;
52 import net.ktnx.mobileledger.model.Data;
53 import net.ktnx.mobileledger.model.LedgerTransaction;
54 import net.ktnx.mobileledger.model.MatchedTemplate;
55 import net.ktnx.mobileledger.ui.FabManager;
56 import net.ktnx.mobileledger.ui.QR;
57 import net.ktnx.mobileledger.ui.activity.ProfileThemedActivity;
58 import net.ktnx.mobileledger.ui.templates.TemplatesActivity;
59 import net.ktnx.mobileledger.utils.Logger;
60 import net.ktnx.mobileledger.utils.MLDB;
61 import net.ktnx.mobileledger.utils.Misc;
63 import java.util.ArrayList;
64 import java.util.List;
65 import java.util.Objects;
66 import java.util.regex.Matcher;
67 import java.util.regex.Pattern;
69 import static net.ktnx.mobileledger.utils.Logger.debug;
71 public class NewTransactionActivity extends ProfileThemedActivity
72 implements TaskCallback, NewTransactionFragment.OnNewTransactionFragmentInteractionListener,
73 QR.QRScanTrigger, QR.QRScanResultReceiver, DescriptionSelectedCallback,
74 FabManager.FabHandler {
75 final String TAG = "new-t-a";
76 private NavController navController;
77 private NewTransactionModel model;
78 private ActivityResultLauncher<Void> qrScanLauncher;
79 private ActivityNewTransactionBinding b;
80 private FabManager fabManager;
82 protected void onCreate(Bundle savedInstanceState) {
83 super.onCreate(savedInstanceState);
85 b = ActivityNewTransactionBinding.inflate(getLayoutInflater(), null, false);
86 setContentView(b.getRoot());
87 setSupportActionBar(b.toolbar);
88 Data.observeProfile(this,
89 mobileLedgerProfile -> b.toolbar.setSubtitle(mobileLedgerProfile.getName()));
91 NavHostFragment navHostFragment = (NavHostFragment) Objects.requireNonNull(
92 getSupportFragmentManager().findFragmentById(R.id.new_transaction_nav));
93 navController = navHostFragment.getNavController();
95 Objects.requireNonNull(getSupportActionBar())
96 .setDisplayHomeAsUpEnabled(true);
98 model = new ViewModelProvider(this).get(NewTransactionModel.class);
100 qrScanLauncher = QR.registerLauncher(this, this);
102 fabManager = new FabManager(b.fabAdd);
104 model.isSubmittable()
105 .observe(this, isSubmittable -> {
107 fabManager.showFab();
110 fabManager.hideFab();
113 // viewModel.checkTransactionSubmittable(listAdapter);
115 b.fabAdd.setOnClickListener(v -> onFabPressed());
118 protected void initProfile() {
119 long profileId = getIntent().getLongExtra(PARAM_PROFILE_ID, 0);
120 int profileHue = getIntent().getIntExtra(PARAM_THEME, -1);
122 if (profileHue < 0) {
123 Logger.debug(TAG, "Started with invalid/missing theme; quitting");
128 if (profileId <= 0) {
129 Logger.debug(TAG, "Started with invalid/missing profile_id; quitting");
134 setupProfileColors(profileHue);
135 initProfile(profileId);
138 public void finish() {
140 overridePendingTransition(R.anim.dummy, R.anim.slide_out_down);
143 public boolean onOptionsItemSelected(MenuItem item) {
144 if (item.getItemId() == android.R.id.home) {
148 return super.onOptionsItemSelected(item);
150 public void onTransactionSave(LedgerTransaction tr) {
151 navController.navigate(R.id.action_newTransactionFragment_to_newTransactionSavingFragment);
154 SendTransactionTask saver =
155 new SendTransactionTask(this, mProfile, model.getSimulateSaveFlag());
158 catch (Exception e) {
159 debug("new-transaction", "Unknown error", e);
161 Bundle b = new Bundle();
162 b.putString("error", "unknown error");
163 navController.navigate(R.id.newTransactionFragment, b);
166 public boolean onSimulateCrashMenuItemClicked(MenuItem item) {
167 debug("crash", "Will crash intentionally");
168 new AsyncCrasher().execute();
171 public boolean onCreateOptionsMenu(Menu menu) {
172 super.onCreateOptionsMenu(menu);
174 if (!BuildConfig.DEBUG)
177 // Inflate the menu; this adds items to the action bar if it is present.
178 getMenuInflater().inflate(R.menu.new_transaction, menu);
180 MenuCompat.setGroupDividerEnabled(menu, true);
182 menu.findItem(R.id.action_simulate_save)
183 .setOnMenuItemClickListener(this::onToggleSimulateSaveMenuItemClicked);
184 menu.findItem(R.id.action_simulate_crash)
185 .setOnMenuItemClickListener(this::onSimulateCrashMenuItemClicked);
187 model.getSimulateSave()
188 .observe(this, state -> {
189 menu.findItem(R.id.action_simulate_save)
191 b.simulationLabel.setVisibility(state ? View.VISIBLE : View.GONE);
198 public int dp2px(float dp) {
199 return Math.round(TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, dp,
200 getResources().getDisplayMetrics()));
203 public void done(String error) {
204 Bundle b = new Bundle();
206 b.putString("error", error);
207 navController.navigate(R.id.action_newTransactionSavingFragment_Failure, b);
210 navController.navigate(R.id.action_newTransactionSavingFragment_Success, b);
212 public boolean onToggleSimulateSaveMenuItemClicked(MenuItem item) {
213 model.toggleSimulateSave();
218 public void triggerQRScan() {
219 qrScanLauncher.launch(null);
221 private void startNewPatternActivity(String scanned) {
222 Intent intent = new Intent(this, TemplatesActivity.class);
223 Bundle args = new Bundle();
224 args.putString(TemplatesActivity.ARG_ADD_TEMPLATE, scanned);
225 startActivity(intent, args);
227 private void alertNoTemplateMatch(String scanned) {
228 MaterialAlertDialogBuilder builder = new MaterialAlertDialogBuilder(this);
229 builder.setCancelable(true)
230 .setMessage(R.string.no_template_matches)
231 .setPositiveButton(R.string.add_button,
232 (dialog, which) -> startNewPatternActivity(scanned))
236 public void onQRScanResult(String text) {
237 Logger.debug("qr", String.format("Got QR scan result [%s]", text));
239 if (Misc.emptyIsNull(text) == null)
242 LiveData<List<TemplateHeader>> allTemplates = DB.get()
245 allTemplates.observe(this, templateHeaders -> {
246 ArrayList<MatchedTemplate> matchingFallbackTemplates = new ArrayList<>();
247 ArrayList<MatchedTemplate> matchingTemplates = new ArrayList<>();
249 for (TemplateHeader ph : templateHeaders) {
250 String patternSource = ph.getRegularExpression();
251 if (Misc.emptyIsNull(patternSource) == null)
254 Pattern pattern = Pattern.compile(patternSource);
255 Matcher matcher = pattern.matcher(text);
256 if (!matcher.matches())
259 Logger.debug("pattern",
260 String.format("Pattern '%s' [%s] matches '%s'", ph.getName(),
261 patternSource, text));
263 matchingFallbackTemplates.add(
264 new MatchedTemplate(ph, matcher.toMatchResult()));
266 matchingTemplates.add(new MatchedTemplate(ph, matcher.toMatchResult()));
268 catch (ParcelFormatException e) {
270 Logger.debug("pattern",
271 String.format("Error compiling regular expression '%s'", patternSource),
276 if (matchingTemplates.isEmpty())
277 matchingTemplates = matchingFallbackTemplates;
279 if (matchingTemplates.isEmpty())
280 alertNoTemplateMatch(text);
281 else if (matchingTemplates.size() == 1)
282 model.applyTemplate(matchingTemplates.get(0), text);
284 chooseTemplate(matchingTemplates, text);
287 private void chooseTemplate(ArrayList<MatchedTemplate> matchingTemplates, String matchedText) {
288 final String templateNameColumn = "name";
289 AbstractCursor cursor = new AbstractCursor() {
291 public int getCount() {
292 return matchingTemplates.size();
295 public String[] getColumnNames() {
296 return new String[]{"_id", templateNameColumn};
299 public String getString(int column) {
301 return String.valueOf(getPosition());
302 return matchingTemplates.get(getPosition()).templateHead.getName();
305 public short getShort(int column) {
307 return (short) getPosition();
311 public int getInt(int column) {
312 return getShort(column);
315 public long getLong(int column) {
316 return getShort(column);
319 public float getFloat(int column) {
320 return getShort(column);
323 public double getDouble(int column) {
324 return getShort(column);
327 public boolean isNull(int column) {
331 public int getColumnCount() {
336 MaterialAlertDialogBuilder builder = new MaterialAlertDialogBuilder(this);
337 builder.setCancelable(true)
338 .setTitle(R.string.choose_template_to_apply)
339 .setIcon(R.drawable.ic_baseline_auto_graph_24)
340 .setSingleChoiceItems(cursor, -1, templateNameColumn, (dialog, which) -> {
341 model.applyTemplate(matchingTemplates.get(which), matchedText);
347 public void descriptionSelected(String description) {
348 debug("description selected", description);
349 if (!model.accountListIsEmpty())
352 String accFilter = mProfile.getPreferredAccountsFilter();
354 ArrayList<String> params = new ArrayList<>();
355 StringBuilder sb = new StringBuilder("select t.profile, t.id from transactions t");
357 if (!TextUtils.isEmpty(accFilter)) {
358 sb.append(" JOIN transaction_accounts ta")
359 .append(" ON ta.profile = t.profile")
360 .append(" AND ta.transaction_id = t.id");
363 sb.append(" WHERE t.description=?");
364 params.add(description);
366 if (!TextUtils.isEmpty(accFilter)) {
367 sb.append(" AND ta.account_name LIKE '%'||?||'%'");
368 params.add(accFilter);
371 sb.append(" ORDER BY t.year desc, t.month desc, t.day desc LIMIT 1");
373 final String sql = sb.toString();
374 debug("description", sql);
375 debug("description", params.toString());
377 // FIXME: handle exceptions?
378 MLDB.queryInBackground(sql, params.toArray(new String[]{}), new MLDB.CallbackHelper() {
380 public void onStart() {
381 model.incrementBusyCounter();
384 public void onDone() {
385 model.decrementBusyCounter();
388 public boolean onRow(@NonNull Cursor cursor) {
389 final long profileId = cursor.getLong(0);
390 final int transactionId = cursor.getInt(1);
391 runOnUiThread(() -> model.loadTransactionIntoModel(profileId, transactionId));
392 return false; // limit 1, by the way
395 public void onNoRows() {
396 if (TextUtils.isEmpty(accFilter))
399 debug("description", "Trying transaction search without preferred account filter");
401 final String broaderSql =
402 "select t.profile, t.id from transactions t where t.description=?" +
403 " ORDER BY year desc, month desc, day desc LIMIT 1";
405 debug("description", broaderSql);
406 debug("description", description);
408 runOnUiThread(() -> Snackbar.make(b.newTransactionNav,
409 R.string.ignoring_preferred_account, Snackbar.LENGTH_INDEFINITE)
412 MLDB.queryInBackground(broaderSql, new String[]{description},
413 new MLDB.CallbackHelper() {
415 public void onStart() {
416 model.incrementBusyCounter();
419 public boolean onRow(@NonNull Cursor cursor) {
420 final long profileId = cursor.getLong(0);
421 final int transactionId = cursor.getInt(1);
422 runOnUiThread(() -> model.loadTransactionIntoModel(profileId,
427 public void onDone() {
428 model.decrementBusyCounter();
434 private void onFabPressed() {
435 fabManager.hideFab();
436 Misc.hideSoftKeyboard(this);
438 LedgerTransaction tr = model.constructLedgerTransaction();
440 onTransactionSave(tr);
443 public Context getContext() {
447 public void showManagedFab() {
448 if (Objects.requireNonNull(model.isSubmittable()
450 fabManager.showFab();
453 public void hideManagedFab() {
454 fabManager.hideFab();