170 lines
5.3 KiB
C++
170 lines
5.3 KiB
C++
#include "Utils.h"
|
|
#include "TDoublyLinkedList.h"
|
|
#include "TStack.h"
|
|
#include "TPerson.h"
|
|
#include <ctime>
|
|
#include <iostream>
|
|
#include <limits>
|
|
|
|
// Displays the main menu and reads user choice.
|
|
int Utils::Choice()
|
|
{
|
|
std::cout << "========\n1. Add line\n2. Remove line\n3. View current1 document\n4. Print queue\n5. Process print job\n6. Undo\n7. Redo\n0. Exit"
|
|
"\n\nChoice: ";
|
|
int choice;
|
|
std::cin >> choice;
|
|
std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
|
|
return choice;
|
|
}
|
|
|
|
// Inserts a new line into the document.
|
|
// Records the action into the undo stack.
|
|
// 'index' determines insert location; defaults to end of document.
|
|
int Utils::Insert(TDoublyLinkedList &document, TStack &undoStack, TStack &redoStack, int index)
|
|
{
|
|
for (int i = 0; i < document.GetSize(); i++) {
|
|
std::cout << i + 1 << ". " << document.GetAtIndex(i) << std::endl;
|
|
}
|
|
if (document.GetSize() > 0)
|
|
{
|
|
std::cout << "Enter the line number where you want to insert the line" <<std::endl;
|
|
if (!(std::cin >> index)) {
|
|
std::cin.clear();
|
|
std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
|
|
std::cout << "========\nIndex must be a number\n========\n\n" << std::endl;
|
|
return index;
|
|
}
|
|
|
|
std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
|
|
}
|
|
if (document.GetSize() < 1)
|
|
index = 1;
|
|
|
|
std::cout << "Enter the text" <<std::endl;
|
|
std::string line;
|
|
std::getline(std::cin, line);
|
|
|
|
|
|
document.InsertAtIndex(index - 1, line);
|
|
undoStack.Push({INSERT, line, index - 1});
|
|
if (!redoStack.IsEmpty()) {
|
|
redoStack.Clear();
|
|
}
|
|
return index;
|
|
}
|
|
|
|
void Utils::PrintList(const TDoublyLinkedList &document)
|
|
{
|
|
for (int i = 0; i < document.GetSize(); i++) {
|
|
std::cout << i + 1 << ". " << document.GetAtIndex(i) << std::endl;
|
|
}
|
|
std::cout << "\n\n";
|
|
}
|
|
|
|
// Removes a line chosen by the user.
|
|
// Action is pushed to undo stack for reversibility.
|
|
int Utils::RemoveLine(TDoublyLinkedList &document, TStack &undoStack, TStack &redoStack, int index)
|
|
{
|
|
std::cout << "Enter the number of the line you want to remove" <<std::endl;
|
|
if (!(std::cin >> index)) {
|
|
std::cin.clear();
|
|
std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
|
|
std::cout << "========\nIndex must be a number\n========\n\n" << std::endl;
|
|
return index;
|
|
} std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
|
|
const std::string deletedLine = document.GetAtIndex(index-1);
|
|
document.Remove(index-1);
|
|
undoStack.Push({DELETE, deletedLine, index-1});
|
|
if (!redoStack.IsEmpty()) {
|
|
redoStack.Clear();
|
|
}
|
|
return index;
|
|
}
|
|
|
|
int Utils::RandomInt(const int min, const int max)
|
|
{
|
|
static bool isSeeded = false;
|
|
if (!isSeeded) {
|
|
std::srand(static_cast<unsigned>(std::time(nullptr))); //<---- not the "best" random seeding available
|
|
isSeeded = true; // but sufficient for this use case
|
|
}
|
|
|
|
if (max <= min)
|
|
return 0;
|
|
|
|
return min + rand() % (max - min + 1); // <---- Limited randomness, but again
|
|
} // sufficient for this use case
|
|
|
|
// Comparison used for cabin grouping (QuickSort):
|
|
// 1) cabinSize ascending
|
|
// 2) lastName alphabetical
|
|
bool Utils::CompareLastnames(const TPerson *a, const TPerson *b)
|
|
{
|
|
if (a->cabinSize < b->cabinSize)
|
|
return true;
|
|
if (a->cabinSize > b->cabinSize)
|
|
return false;
|
|
return a->lastName < b->lastName;
|
|
}
|
|
|
|
|
|
int Utils::Partition(TPerson **arr, const int startIndex, const int endIndex)
|
|
{
|
|
TPerson *pivot = arr[endIndex];
|
|
int i = startIndex - 1;
|
|
|
|
for (int j = startIndex; j < endIndex; j++) {
|
|
if (CompareLastnames(arr[j], pivot)) {
|
|
i++;
|
|
std::swap(arr[i], arr[j]);
|
|
}
|
|
}
|
|
|
|
|
|
std::swap(arr[i + 1], arr[endIndex]);
|
|
return i + 1;
|
|
}
|
|
|
|
// QuickSort on an array of TPerson* using CompareLastnames:
|
|
// - Average time: O(n log n)
|
|
// - Worst case: O(n^2) if pivot choices are bad
|
|
// - Sorts in-place (no extra arrays)
|
|
void Utils::QuickSort(TPerson** arr, const int low, const int high)
|
|
{
|
|
if (low < high) {
|
|
int p = Partition(arr, low, high);
|
|
QuickSort(arr, low, p - 1);
|
|
QuickSort(arr, p + 1, high);
|
|
}
|
|
}
|
|
|
|
// Binary search on an alphabetically sorted array of TPerson* (by lastName, then firstName).
|
|
// Primary search key: surname. If no surname match is found, falls back to linear scan on firstName.
|
|
int Utils::BinarySearch(TPerson** arr, int p1, int p2, const std::string &target)
|
|
{
|
|
const int origStart = p1;
|
|
const int origEnd = p2;
|
|
|
|
|
|
while (p1 <= p2) {
|
|
const int newP = (p1 + p2) / 2;
|
|
std::string currentFirst = arr[newP]->firstName;
|
|
std::string currentLast = arr[newP]->lastName;
|
|
|
|
if (target == currentFirst || target == currentLast)
|
|
return newP;
|
|
|
|
if (target > currentLast)
|
|
p1 = newP + 1;
|
|
|
|
else
|
|
p2 = newP - 1;
|
|
}
|
|
|
|
// Fallback linear scan for first names if no last name match
|
|
for (int i = origStart; i <= origEnd; i++) {
|
|
if (arr[i]->firstName == target)
|
|
return i;
|
|
}
|
|
return -1;
|
|
} |