<?php

require_once __DIR__.'/Base.php';

use Model\User;
use Model\Timetable;
use Model\TimetableDay;
use Model\TimetableWeek;
use Model\TimetableOff;
use Model\TimetableExtra;

class TimetableTest extends Base
{
    public function testCalculateWorkDays()
    {
        $w = new TimetableWeek($this->container);
        $t = new Timetable($this->container);

        $this->assertNotFalse($w->create(1, 1, '09:30', '12:00'));
        $this->assertNotFalse($w->create(1, 1, '13:00', '17:00'));
        $this->assertNotFalse($w->create(1, 2, '09:30', '12:00'));
        $this->assertNotFalse($w->create(1, 2, '13:00', '17:00'));

        $monday = new DateTime('next Monday');

        $timetable = $t->calculate(1, $monday, new DateTime('next Monday + 6 days'));
        $this->assertNotEmpty($timetable);
        $this->assertCount(4, $timetable);

        $this->assertEquals($monday->format('Y-m-d').' 09:30', $timetable[0][0]->format('Y-m-d H:i'));
        $this->assertEquals($monday->format('Y-m-d').' 12:00', $timetable[0][1]->format('Y-m-d H:i'));
        $this->assertEquals($monday->format('Y-m-d').' 13:00', $timetable[1][0]->format('Y-m-d H:i'));
        $this->assertEquals($monday->format('Y-m-d').' 17:00', $timetable[1][1]->format('Y-m-d H:i'));

        $this->assertEquals($monday->add(new DateInterval('P1D'))->format('Y-m-d').' 09:30', $timetable[2][0]->format('Y-m-d H:i'));
        $this->assertEquals($monday->format('Y-m-d').' 12:00', $timetable[2][1]->format('Y-m-d H:i'));
        $this->assertEquals($monday->format('Y-m-d').' 13:00', $timetable[3][0]->format('Y-m-d H:i'));
        $this->assertEquals($monday->format('Y-m-d').' 17:00', $timetable[3][1]->format('Y-m-d H:i'));
    }

    public function testCalculateOverTime()
    {
        $d = new TimetableDay($this->container);
        $w = new TimetableWeek($this->container);
        $e = new TimetableExtra($this->container);
        $t = new Timetable($this->container);

        $monday = new DateTime('next Monday');
        $tuesday = new DateTime('next Monday + 1 day');
        $friday = new DateTime('next Monday + 4 days');

        $this->assertNotFalse($d->create(1, '08:00', '12:00'));
        $this->assertNotFalse($d->create(1, '14:00', '18:00'));

        $this->assertNotFalse($w->create(1, 1, '09:30', '12:00'));
        $this->assertNotFalse($w->create(1, 1, '13:00', '17:00'));
        $this->assertNotFalse($w->create(1, 2, '09:30', '12:00'));
        $this->assertNotFalse($w->create(1, 2, '13:00', '17:00'));

        $this->assertNotFalse($e->create(1, $tuesday->format('Y-m-d'), 0, '17:00', '22:00'));
        $this->assertNotFalse($e->create(1, $friday->format('Y-m-d'), 1));

        $timetable = $t->calculate(1, $monday, new DateTime('next Monday + 6 days'));
        $this->assertNotEmpty($timetable);
        $this->assertCount(7, $timetable);

        $this->assertEquals($monday->format('Y-m-d').' 09:30', $timetable[0][0]->format('Y-m-d H:i'));
        $this->assertEquals($monday->format('Y-m-d').' 12:00', $timetable[0][1]->format('Y-m-d H:i'));
        $this->assertEquals($monday->format('Y-m-d').' 13:00', $timetable[1][0]->format('Y-m-d H:i'));
        $this->assertEquals($monday->format('Y-m-d').' 17:00', $timetable[1][1]->format('Y-m-d H:i'));

        $this->assertEquals($tuesday->format('Y-m-d').' 09:30', $timetable[2][0]->format('Y-m-d H:i'));
        $this->assertEquals($tuesday->format('Y-m-d').' 12:00', $timetable[2][1]->format('Y-m-d H:i'));
        $this->assertEquals($tuesday->format('Y-m-d').' 13:00', $timetable[3][0]->format('Y-m-d H:i'));
        $this->assertEquals($tuesday->format('Y-m-d').' 17:00', $timetable[3][1]->format('Y-m-d H:i'));

        $this->assertEquals($tuesday->format('Y-m-d').' 17:00', $timetable[4][0]->format('Y-m-d H:i'));
        $this->assertEquals($tuesday->format('Y-m-d').' 22:00', $timetable[4][1]->format('Y-m-d H:i'));

        $this->assertEquals($friday->format('Y-m-d').' 08:00', $timetable[5][0]->format('Y-m-d H:i'));
        $this->assertEquals($friday->format('Y-m-d').' 12:00', $timetable[5][1]->format('Y-m-d H:i'));

        $this->assertEquals($friday->format('Y-m-d').' 14:00', $timetable[6][0]->format('Y-m-d H:i'));
        $this->assertEquals($friday->format('Y-m-d').' 18:00', $timetable[6][1]->format('Y-m-d H:i'));
    }

    public function testCalculateTimeOff()
    {
        $d = new TimetableDay($this->container);
        $w = new TimetableWeek($this->container);
        $o = new TimetableOff($this->container);
        $t = new Timetable($this->container);

        $monday = new DateTime('next Monday');
        $tuesday = new DateTime('next Monday + 1 day');
        $friday = new DateTime('next Monday + 4 days');

        $this->assertNotFalse($d->create(1, '08:00', '12:00'));
        $this->assertNotFalse($d->create(1, '14:00', '18:00'));

        $this->assertNotFalse($w->create(1, 1, '09:30', '12:00'));
        $this->assertNotFalse($w->create(1, 1, '13:00', '17:00'));
        $this->assertNotFalse($w->create(1, 2, '09:30', '12:00'));
        $this->assertNotFalse($w->create(1, 2, '13:00', '17:00'));
        $this->assertNotFalse($w->create(1, 5, '09:30', '12:00'));
        $this->assertNotFalse($w->create(1, 5, '13:00', '17:00'));

        $this->assertNotFalse($o->create(1, $tuesday->format('Y-m-d'), 0, '14:00', '15:00'));
        $this->assertNotFalse($o->create(1, $monday->format('Y-m-d'), 1));

        $timetable = $t->calculate(1, $monday, new DateTime('next Monday + 6 days'));
        $this->assertNotEmpty($timetable);
        $this->assertCount(5, $timetable);

        $this->assertEquals($tuesday->format('Y-m-d').' 09:30', $timetable[0][0]->format('Y-m-d H:i'));
        $this->assertEquals($tuesday->format('Y-m-d').' 12:00', $timetable[0][1]->format('Y-m-d H:i'));

        $this->assertEquals($tuesday->format('Y-m-d').' 13:00', $timetable[1][0]->format('Y-m-d H:i'));
        $this->assertEquals($tuesday->format('Y-m-d').' 14:00', $timetable[1][1]->format('Y-m-d H:i'));

        $this->assertEquals($tuesday->format('Y-m-d').' 15:00', $timetable[2][0]->format('Y-m-d H:i'));
        $this->assertEquals($tuesday->format('Y-m-d').' 17:00', $timetable[2][1]->format('Y-m-d H:i'));

        $this->assertEquals($friday->format('Y-m-d').' 09:30', $timetable[3][0]->format('Y-m-d H:i'));
        $this->assertEquals($friday->format('Y-m-d').' 12:00', $timetable[3][1]->format('Y-m-d H:i'));

        $this->assertEquals($friday->format('Y-m-d').' 13:00', $timetable[4][0]->format('Y-m-d H:i'));
        $this->assertEquals($friday->format('Y-m-d').' 17:00', $timetable[4][1]->format('Y-m-d H:i'));
    }

    public function testClosestTimeSlot()
    {
        $w = new TimetableWeek($this->container);
        $t = new Timetable($this->container);

        $this->assertNotFalse($w->create(1, 1, '09:30', '12:00'));
        $this->assertNotFalse($w->create(1, 1, '13:00', '17:00'));
        $this->assertNotFalse($w->create(1, 2, '09:30', '12:00'));
        $this->assertNotFalse($w->create(1, 2, '13:00', '17:00'));

        $monday = new DateTime('Monday');
        $tuesday = new DateTime('Tuesday');

        $timetable = $t->calculate(1, new DateTime('Monday'), new DateTime('Monday + 6 days'));
        $this->assertNotEmpty($timetable);
        $this->assertCount(4, $timetable);

        // Start to work before timetable
        $date = new DateTime('Monday');
        $date->setTime(5, 02);

        $slot = $t->findClosestTimeSlot($date, $timetable);
        $this->assertNotEmpty($slot);
        $this->assertEquals($monday->format('Y-m-d').' 09:30', $slot[0]->format('Y-m-d H:i'));
        $this->assertEquals($monday->format('Y-m-d').' 12:00', $slot[1]->format('Y-m-d H:i'));

        // Start to work at the end of the timeslot
        $date = new DateTime('Monday');
        $date->setTime(12, 02);

        $slot = $t->findClosestTimeSlot($date, $timetable);
        $this->assertNotEmpty($slot);
        $this->assertEquals($monday->format('Y-m-d').' 09:30', $slot[0]->format('Y-m-d H:i'));
        $this->assertEquals($monday->format('Y-m-d').' 12:00', $slot[1]->format('Y-m-d H:i'));

        // Start to work at lunch time
        $date = new DateTime('Monday');
        $date->setTime(12, 32);

        $slot = $t->findClosestTimeSlot($date, $timetable);
        $this->assertNotEmpty($slot);
        $this->assertEquals($monday->format('Y-m-d').' 13:00', $slot[0]->format('Y-m-d H:i'));
        $this->assertEquals($monday->format('Y-m-d').' 17:00', $slot[1]->format('Y-m-d H:i'));

        // Start to work early in the morning
        $date = new DateTime('Tuesday');
        $date->setTime(8, 02);

        $slot = $t->findClosestTimeSlot($date, $timetable);
        $this->assertNotEmpty($slot);
        $this->assertEquals($tuesday->format('Y-m-d').' 09:30', $slot[0]->format('Y-m-d H:i'));
        $this->assertEquals($tuesday->format('Y-m-d').' 12:00', $slot[1]->format('Y-m-d H:i'));
    }

    public function testCalculateDuration()
    {
        $w = new TimetableWeek($this->container);
        $t = new Timetable($this->container);

        $this->assertNotFalse($w->create(1, 1, '09:30', '12:00'));
        $this->assertNotFalse($w->create(1, 1, '13:00', '17:00'));
        $this->assertNotFalse($w->create(1, 2, '09:30', '12:00'));
        $this->assertNotFalse($w->create(1, 2, '13:00', '17:00'));

        // Different day
        $start = new DateTime('Monday');
        $start->setTime(16, 02);

        $end = new DateTime('Tuesday');
        $end->setTime(10, 03);

        $this->assertEquals(1.5, $t->calculateEffectiveDuration(1, $start, $end));

        // Same time slot
        $start = new DateTime('Monday');
        $start->setTime(16, 02);

        $end = new DateTime('Monday');
        $end->setTime(17, 03);

        $this->assertEquals(1, $t->calculateEffectiveDuration(1, $start, $end));

        // Intermediate time slot
        $start = new DateTime('Monday');
        $start->setTime(10, 02);

        $end = new DateTime('Tuesday');
        $end->setTime(16, 03);

        $this->assertEquals(11.5, $t->calculateEffectiveDuration(1, $start, $end));

        // Different day
        $start = new DateTime('Monday');
        $start->setTime(9, 02);

        $end = new DateTime('Tuesday');
        $end->setTime(10, 03);

        $this->assertEquals(7, $t->calculateEffectiveDuration(1, $start, $end));

        // Start before first time slot
        $start = new DateTime('Monday');
        $start->setTime(5, 32);

        $end = new DateTime('Tuesday');
        $end->setTime(11, 17);

        $this->assertEquals(8.25, $t->calculateEffectiveDuration(1, $start, $end));
    }

    public function testCalculateDurationWithEmptyTimetable()
    {
        $t = new Timetable($this->container);

        $start = new DateTime('Monday');
        $start->setTime(16, 02);

        $end = new DateTime('Monday');
        $end->setTime(17, 03);

        $this->assertEquals(1, $t->calculateEffectiveDuration(1, $start, $end));
    }
}