mirror of
				https://github.com/subsurface/subsurface.git
				synced 2025-02-19 22:16:15 +00:00 
			
		
		
		
	This causes UI confusion. Notably we go into edit mode and reduce the number of samples, leading to loss of information. If someone really manually adds a dive with more than 50 samples, they should still be able to explicitly open the dive in the planner. Signed-off-by: Berthold Stoeger <bstoeger@mail.tuwien.ac.at>
		
			
				
	
	
		
			419 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			419 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0
 | |
| #include "TabDiveNotes.h"
 | |
| #include "core/divesite.h"
 | |
| #include "core/qthelper.h"
 | |
| #include "core/selection.h"
 | |
| #include "core/subsurface-string.h"
 | |
| #include "core/trip.h"
 | |
| #include "desktop-widgets/mainwindow.h"
 | |
| #include "desktop-widgets/mapwidget.h"
 | |
| #include "desktop-widgets/simplewidgets.h"
 | |
| #include "qt-models/diveplannermodel.h"
 | |
| #include "commands/command.h"
 | |
| 
 | |
| #include <QCompleter>
 | |
| #include <QMessageBox>
 | |
| 
 | |
| struct Completers {
 | |
| 	QCompleter *diveguide;
 | |
| 	QCompleter *buddy;
 | |
| 	QCompleter *tags;
 | |
| };
 | |
| 
 | |
| TabDiveNotes::TabDiveNotes(QWidget *parent) : TabBase(parent),
 | |
| 	ignoreInput(false),
 | |
| 	currentTrip(0)
 | |
| {
 | |
| 	ui.setupUi(this);
 | |
| 
 | |
| 	updateDateTimeFields();
 | |
| 
 | |
| 	connect(&diveListNotifier, &DiveListNotifier::divesChanged, this, &TabDiveNotes::divesChanged);
 | |
| 	connect(&diveListNotifier, &DiveListNotifier::tripChanged, this, &TabDiveNotes::tripChanged);
 | |
| 	connect(&diveListNotifier, &DiveListNotifier::diveSiteChanged, this, &TabDiveNotes::diveSiteEdited);
 | |
| 	connect(&diveListNotifier, &DiveListNotifier::commandExecuted, this, &TabDiveNotes::closeWarning);
 | |
| 
 | |
| 	connect(ui.editDiveSiteButton, &QToolButton::clicked, MainWindow::instance(), &MainWindow::startDiveSiteEdit);
 | |
| #ifdef MAP_SUPPORT
 | |
| 	connect(ui.location, &DiveLocationLineEdit::entered, MapWidget::instance(), &MapWidget::centerOnIndex);
 | |
| 	connect(ui.location, &DiveLocationLineEdit::currentChanged, MapWidget::instance(), &MapWidget::centerOnIndex);
 | |
| #endif
 | |
| 	connect(ui.location, &DiveLocationLineEdit::editingFinished, this, &TabDiveNotes::on_location_diveSiteSelected);
 | |
| 
 | |
| 	// One might think that we could listen to the precise property-changed signals of the preferences system.
 | |
| 	// Alas, this is not the case. When the user switches to system-format, the preferences sends the according
 | |
| 	// signal. However, the correct date and time format is set by the preferences dialog later. This should be fixed.
 | |
| 	connect(&diveListNotifier, &DiveListNotifier::settingsChanged, this, &TabDiveNotes::updateDateTimeFields);
 | |
| 
 | |
| 	QAction *action = new QAction(tr("OK"), this);
 | |
| 	connect(action, &QAction::triggered, this, &TabDiveNotes::closeWarning);
 | |
| 	ui.multiDiveWarningMessage->addAction(action);
 | |
| 
 | |
| 	action = new QAction(tr("Undo"), this);
 | |
| 	connect(action, &QAction::triggered, Command::undoAction(this), &QAction::trigger);
 | |
| 	connect(action, &QAction::triggered, this, &TabDiveNotes::closeWarning);
 | |
| 	ui.multiDiveWarningMessage->addAction(action);
 | |
| 
 | |
| 	Completers completers;
 | |
| 	completers.buddy = new QCompleter(&buddyModel, ui.buddy);
 | |
| 	completers.diveguide = new QCompleter(&diveGuideModel, ui.diveguide);
 | |
| 	completers.tags = new QCompleter(&tagModel, ui.tagWidget);
 | |
| 	completers.buddy->setCaseSensitivity(Qt::CaseInsensitive);
 | |
| 	completers.diveguide->setCaseSensitivity(Qt::CaseInsensitive);
 | |
| 	completers.tags->setCaseSensitivity(Qt::CaseInsensitive);
 | |
| 	ui.buddy->setCompleter(completers.buddy);
 | |
| 	ui.diveguide->setCompleter(completers.diveguide);
 | |
| 	ui.tagWidget->setCompleter(completers.tags);
 | |
| 	ui.multiDiveWarningMessage->hide();
 | |
| 	ui.depth->hide();
 | |
| 	ui.depthLabel->hide();
 | |
| 	ui.duration->hide();
 | |
| 	ui.durationLabel->hide();
 | |
| 	setMinimumHeight(0);
 | |
| 	setMinimumWidth(0);
 | |
| 
 | |
| 	connect(ui.multiDiveWarningMessage, &KMessageWidget::showAnimationFinished,
 | |
| 					ui.location, &DiveLocationLineEdit::fixPopupPosition);
 | |
| 
 | |
| 	// enable URL clickability in notes:
 | |
| 	new TextHyperlinkEventFilter(ui.notes); //destroyed when ui.notes is destroyed
 | |
| 
 | |
| 	ui.diveTripLocation->hide();
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::updateDateTimeFields()
 | |
| {
 | |
| 	ui.dateEdit->setDisplayFormat(prefs.date_format);
 | |
| 	ui.timeEdit->setDisplayFormat(prefs.time_format);
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::closeWarning()
 | |
| {
 | |
| 	ui.multiDiveWarningMessage->hide();
 | |
| }
 | |
| 
 | |
| // This function gets called if a field gets updated by an undo command.
 | |
| // Refresh the corresponding UI field.
 | |
| void TabDiveNotes::divesChanged(const QVector<dive *> &dives, DiveField field)
 | |
| {
 | |
| 	// If the current dive is not in list of changed dives, do nothing
 | |
| 	if (!current_dive || !dives.contains(current_dive))
 | |
| 		return;
 | |
| 
 | |
| 	if (field.duration)
 | |
| 		ui.duration->setText(render_seconds_to_string(current_dive->duration.seconds));
 | |
| 	if (field.depth)
 | |
| 		ui.depth->setText(get_depth_string(current_dive->maxdepth, true));
 | |
| 	if (field.rating)
 | |
| 		ui.rating->setCurrentStars(current_dive->rating);
 | |
| 	if (field.notes)
 | |
| 		updateNotes(current_dive);
 | |
| 	if (field.datetime) {
 | |
| 		updateDateTime(current_dive);
 | |
| 		DivePlannerPointsModel::instance()->getDiveplan().when = current_dive->when;
 | |
| 	}
 | |
| 	if (field.divesite)
 | |
| 		updateDiveSite(current_dive);
 | |
| 	if (field.tags)
 | |
| 		ui.tagWidget->setText(get_taglist_string(current_dive->tag_list));
 | |
| 	if (field.buddy)
 | |
| 		ui.buddy->setText(current_dive->buddy);
 | |
| 	if (field.diveguide)
 | |
| 		ui.diveguide->setText(current_dive->diveguide);
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::diveSiteEdited(dive_site *ds, int)
 | |
| {
 | |
| 	if (current_dive && current_dive->dive_site == ds)
 | |
| 		updateDiveSite(current_dive);
 | |
| }
 | |
| 
 | |
| // This function gets called if a trip-field gets updated by an undo command.
 | |
| // Refresh the corresponding UI field.
 | |
| void TabDiveNotes::tripChanged(dive_trip *trip, TripField field)
 | |
| {
 | |
| 	// If the current dive is not in list of changed dives, do nothing
 | |
| 	if (currentTrip != trip)
 | |
| 		return;
 | |
| 
 | |
| 	if (field.notes)
 | |
| 		ui.notes->setText(currentTrip->notes);
 | |
| 	if (field.location)
 | |
| 		ui.diveTripLocation->setText(currentTrip->location);
 | |
| }
 | |
| 
 | |
| static bool isHtml(const QString &s)
 | |
| {
 | |
| 	return s.contains("<div", Qt::CaseInsensitive) || s.contains("<table", Qt::CaseInsensitive);
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::updateNotes(const struct dive *d)
 | |
| {
 | |
| 	QString tmp(d->notes);
 | |
| 	if (isHtml(tmp)) {
 | |
| 		ui.notes->setHtml(tmp);
 | |
| 	} else {
 | |
| 		ui.notes->setPlainText(tmp);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::updateDateTime(const struct dive *d)
 | |
| {
 | |
| 	QDateTime localTime = timestampToDateTime(d->when);
 | |
| 	ui.dateEdit->setDate(localTime.date());
 | |
| 	ui.timeEdit->setTime(localTime.time());
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::updateTripDate(const struct dive_trip *t)
 | |
| {
 | |
| 	QDateTime localTime = timestampToDateTime(trip_date(t));
 | |
| 	ui.dateEdit->setDate(localTime.date());
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::updateDiveSite(struct dive *d)
 | |
| {
 | |
| 	struct dive_site *ds = d->dive_site;
 | |
| 	ui.location->setCurrentDiveSite(d);
 | |
| 	if (ds) {
 | |
| 		ui.locationTags->setText(constructLocationTags(&ds->taxonomy, true));
 | |
| 
 | |
| 		if (ui.locationTags->text().isEmpty() && has_location(&ds->location))
 | |
| 			ui.locationTags->setText(printGPSCoords(&ds->location));
 | |
| 		ui.editDiveSiteButton->setEnabled(true);
 | |
| 	} else {
 | |
| 		ui.locationTags->clear();
 | |
| 		ui.editDiveSiteButton->setEnabled(false);
 | |
| 	}
 | |
| 
 | |
| 	if (ui.locationTags->text().isEmpty())
 | |
| 		ui.locationTags->hide();
 | |
| 	else
 | |
| 		ui.locationTags->show();
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::updateData()
 | |
| {
 | |
| 	ui.location->refreshDiveSiteCache();
 | |
| 
 | |
| 	ignoreInput = true; // don't trigger on changes to the widgets
 | |
| 
 | |
| 	currentTrip = single_selected_trip();
 | |
| 	if (currentTrip) {
 | |
| 		// only use trip relevant fields
 | |
| 		ui.diveguide->setVisible(false);
 | |
| 		ui.DiveguideLabel->setVisible(false);
 | |
| 		ui.buddy->setVisible(false);
 | |
| 		ui.BuddyLabel->setVisible(false);
 | |
| 		ui.rating->setVisible(false);
 | |
| 		ui.RatingLabel->setVisible(false);
 | |
| 		ui.tagWidget->setVisible(false);
 | |
| 		ui.TagLabel->setVisible(false);
 | |
| 		ui.dateEdit->setReadOnly(true);
 | |
| 		ui.timeLabel->setVisible(false);
 | |
| 		ui.timeEdit->setVisible(false);
 | |
| 		ui.diveTripLocation->show();
 | |
| 		ui.location->hide();
 | |
| 		ui.locationPopupButton->hide();
 | |
| 		ui.editDiveSiteButton->hide();
 | |
| 		// rename the remaining fields and fill data from selected trip
 | |
| 		ui.LocationLabel->setText(tr("Trip location"));
 | |
| 		ui.diveTripLocation->setText(currentTrip->location);
 | |
| 		updateTripDate(currentTrip);
 | |
| 		ui.locationTags->clear();
 | |
| 		//TODO: Fix this.
 | |
| 		//ui.location->setText(currentTrip->location);
 | |
| 		ui.NotesLabel->setText(tr("Trip notes"));
 | |
| 		ui.notes->setText(currentTrip->notes);
 | |
| 		ui.depth->setVisible(false);
 | |
| 		ui.depthLabel->setVisible(false);
 | |
| 		ui.duration->setVisible(false);
 | |
| 		ui.durationLabel->setVisible(false);
 | |
| 	} else {
 | |
| 		// make all the fields visible writeable
 | |
| 		ui.diveTripLocation->hide();
 | |
| 		ui.location->show();
 | |
| 		ui.locationPopupButton->show();
 | |
| 		ui.editDiveSiteButton->show();
 | |
| 		ui.diveguide->setVisible(true);
 | |
| 		ui.buddy->setVisible(true);
 | |
| 		ui.rating->setVisible(true);
 | |
| 		ui.RatingLabel->setVisible(true);
 | |
| 		ui.BuddyLabel->setVisible(true);
 | |
| 		ui.DiveguideLabel->setVisible(true);
 | |
| 		ui.TagLabel->setVisible(true);
 | |
| 		ui.tagWidget->setVisible(true);
 | |
| 		ui.dateEdit->setReadOnly(false);
 | |
| 		ui.timeLabel->setVisible(true);
 | |
| 		ui.timeEdit->setVisible(true);
 | |
| 		/* and fill them from the dive */
 | |
| 		ui.rating->setCurrentStars(current_dive->rating);
 | |
| 		// reset labels in case we last displayed trip notes
 | |
| 		ui.LocationLabel->setText(tr("Location"));
 | |
| 		ui.NotesLabel->setText(tr("Notes"));
 | |
| 		ui.tagWidget->setText(get_taglist_string(current_dive->tag_list));
 | |
| 		bool isManual = is_manually_added_dc(¤t_dive->dc);
 | |
| 		ui.depth->setVisible(isManual);
 | |
| 		ui.depthLabel->setVisible(isManual);
 | |
| 		ui.duration->setVisible(isManual);
 | |
| 		ui.durationLabel->setVisible(isManual);
 | |
| 
 | |
| 		updateNotes(current_dive);
 | |
| 		updateDiveSite(current_dive);
 | |
| 		updateDateTime(current_dive);
 | |
| 		ui.diveguide->setText(current_dive->diveguide);
 | |
| 		ui.buddy->setText(current_dive->buddy);
 | |
| 	}
 | |
| 	ui.duration->setText(render_seconds_to_string(current_dive->duration.seconds));
 | |
| 	ui.depth->setText(get_depth_string(current_dive->maxdepth, true));
 | |
| 
 | |
| 	ui.editDiveSiteButton->setEnabled(!ui.location->text().isEmpty());
 | |
| 	/* unset the special value text for date and time, just in case someone dove at midnight */
 | |
| 	ui.dateEdit->setSpecialValueText(QString());
 | |
| 	ui.timeEdit->setSpecialValueText(QString());
 | |
| 
 | |
| 	ignoreInput = false;
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::clear()
 | |
| {
 | |
| 	ui.rating->setCurrentStars(0);
 | |
| 	ui.location->clear();
 | |
| 	ui.diveguide->clear();
 | |
| 	ui.buddy->clear();
 | |
| 	ui.notes->clear();
 | |
| 	/* set date and time to minimums which triggers showing the special value text */
 | |
| 	ui.dateEdit->setSpecialValueText(QString("-"));
 | |
| 	ui.dateEdit->setMinimumDate(QDate(1, 1, 1));
 | |
| 	ui.dateEdit->setDate(QDate(1, 1, 1));
 | |
| 	ui.timeEdit->setSpecialValueText(QString("-"));
 | |
| 	ui.timeEdit->setMinimumTime(QTime(0, 0, 0, 0));
 | |
| 	ui.timeEdit->setTime(QTime(0, 0, 0, 0));
 | |
| 	ui.tagWidget->clear();
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::divesEdited(int i)
 | |
| {
 | |
| 	// No warning if only one dive was edited
 | |
| 	if (i <= 1)
 | |
| 		return;
 | |
| 	ui.multiDiveWarningMessage->setCloseButtonVisible(false);
 | |
| 	ui.multiDiveWarningMessage->setText(tr("Warning: edited %1 dives").arg(i));
 | |
| 	ui.multiDiveWarningMessage->show();
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::on_buddy_editingFinished()
 | |
| {
 | |
| 	if (ignoreInput || !current_dive)
 | |
| 		return;
 | |
| 
 | |
| 	divesEdited(Command::editBuddies(stringToList(ui.buddy->toPlainText()), false));
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::on_diveguide_editingFinished()
 | |
| {
 | |
| 	if (ignoreInput || !current_dive)
 | |
| 		return;
 | |
| 
 | |
| 	divesEdited(Command::editDiveGuide(stringToList(ui.diveguide->toPlainText()), false));
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::on_duration_editingFinished()
 | |
| {
 | |
| 	if (ignoreInput || !current_dive)
 | |
| 		return;
 | |
| 
 | |
| 	// Duration editing is special: we only edit the current dive.
 | |
| 	divesEdited(Command::editDuration(parseDurationToSeconds(ui.duration->text()), true));
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::on_depth_editingFinished()
 | |
| {
 | |
| 	if (ignoreInput || !current_dive)
 | |
| 		return;
 | |
| 
 | |
| 	// Depth editing is special: we only edit the current dive.
 | |
| 	divesEdited(Command::editDepth(parseLengthToMm(ui.depth->text()), true));
 | |
| }
 | |
| 
 | |
| // Editing of the dive time is different. If multiple dives are edited,
 | |
| // all dives are shifted by an offset.
 | |
| static void shiftTime(QDateTime &dateTime)
 | |
| {
 | |
| 	timestamp_t when = dateTimeToTimestamp(dateTime);
 | |
| 	if (current_dive && current_dive->when != when) {
 | |
| 		timestamp_t offset = when - current_dive->when;
 | |
| 		Command::shiftTime(getDiveSelection(), (int)offset);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::on_dateEdit_editingFinished()
 | |
| {
 | |
| 	if (ignoreInput || !current_dive)
 | |
| 		return;
 | |
| 	QDateTime dateTime = timestampToDateTime(current_dive->when);
 | |
| 	dateTime.setDate(ui.dateEdit->date());
 | |
| 	shiftTime(dateTime);
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::on_timeEdit_editingFinished()
 | |
| {
 | |
| 	if (ignoreInput || !current_dive)
 | |
| 		return;
 | |
| 	QDateTime dateTime = timestampToDateTime(current_dive->when);
 | |
| 	dateTime.setTime(ui.timeEdit->time());
 | |
| 	shiftTime(dateTime);
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::on_tagWidget_editingFinished()
 | |
| {
 | |
| 	if (ignoreInput || !current_dive)
 | |
| 		return;
 | |
| 
 | |
| 	divesEdited(Command::editTags(ui.tagWidget->getBlockStringList(), false));
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::on_location_diveSiteSelected()
 | |
| {
 | |
| 	if (ignoreInput || !current_dive)
 | |
| 		return;
 | |
| 
 | |
| 	struct dive_site *newDs = ui.location->currDiveSite();
 | |
| 	if (newDs == RECENTLY_ADDED_DIVESITE)
 | |
| 		divesEdited(Command::editDiveSiteNew(ui.location->text(), false));
 | |
| 	else
 | |
| 		divesEdited(Command::editDiveSite(newDs, false));
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::on_locationPopupButton_clicked()
 | |
| {
 | |
| 	ui.location->showAllSites();
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::on_diveTripLocation_editingFinished()
 | |
| {
 | |
| 	if (!currentTrip)
 | |
| 		return;
 | |
| 	Command::editTripLocation(currentTrip, ui.diveTripLocation->text());
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::on_notes_editingFinished()
 | |
| {
 | |
| 	if (!currentTrip && !current_dive)
 | |
| 		return;
 | |
| 
 | |
| 	QString html = ui.notes->toHtml();
 | |
| 	QString notes = isHtml(html) ? html : ui.notes->toPlainText();
 | |
| 
 | |
| 	if (currentTrip)
 | |
| 		Command::editTripNotes(currentTrip, notes);
 | |
| 	else
 | |
| 		divesEdited(Command::editNotes(notes, false));
 | |
| }
 | |
| 
 | |
| void TabDiveNotes::on_rating_valueChanged(int value)
 | |
| {
 | |
| 	if (ignoreInput || !current_dive)
 | |
| 		return;
 | |
| 
 | |
| 	divesEdited(Command::editRating(value, false));
 | |
| }
 |