<?php
    
    error_reporting(-1);
    mb_internal_encoding('utf-8');
  
    define("SMALL_COL", 10);
    define("MEDIUM_COL", 20);
    define("LARGE_COL", 25);
  
    class Company
    {
        private $name; //Имя компанию
        private $listOfDepatments; //Список департаментов
        
        public function __construct($name = 'Без имени')
        {
            $this->name = $name;
            $this->listOfDepatments = array();
        }

        //Получить имя компании
        public function getNameCompany()
        {
            return $this->name;
        }
        
        //Количество департаментов
        public function getCountDeparments()
        {
            return count($this->listOfDepatments);
        }
        
        //Добавить новый департамент
        public function addDepartmentToCompany($newDepartment)
        {
            $this->listOfDepatments[] = $newDepartment;
        }
        
        //Распечатать информацию о компании (стоит ли вынести это в отдельный класс (например, CompanyReport??))
        public function printInfoAboutCompany()
        {
            $countDepartment = $this->getCountDeparments();
            $total = array('сотр' => 0, 'тугр' => 0, 'кофе' => 0, 'стр' => 0);

            echo "Отчет компании \"{$this->getNameCompany()}\"";
            echo padRigth("Департамент", LARGE_COL) . padLeft("сотр.", SMALL_COL) . padLeft("тугр.", MEDIUM_COL) .
                 padLeft("кофе", SMALL_COL) . padLeft("стр.", SMALL_COL) . padLeft("тугр./стр.", MEDIUM_COL) . "\n";
            foreach ($this->listOfDepatments as $departament) {
                echo padRigth($departament->getDepartmentName(), LARGE_COL) . padLeft($departament->getCountEmployee(), SMALL_COL) .
                     padLeft($departament->getTotalSalary(), MEDIUM_COL) . padLeft($departament->getTotalCups(), SMALL_COL) .
                     padLeft($departament->getTotalPages(), SMALL_COL) . padLeft($departament->getTotalSalary() / $departament->getTotalPages(), MEDIUM_COL) . "\n";
                $total['сотр'] += $departament->getCountEmployee();
                $total['тугр'] += $departament->getTotalSalary();
                $total['кофе'] += $departament->getTotalCups();
                $total['стр'] += $departament->getTotalPages();
            }
            
            //возможно не правильно расчитываю среднее значение
            echo padRigth('Среднее', LARGE_COL) . padLeft($total['сотр'] / $countDepartment, SMALL_COL) .
                     padLeft($total['тугр'] / $countDepartment, MEDIUM_COL) . padLeft($total['кофе'] / $countDepartment, SMALL_COL) .
                     padLeft($total['стр'] / $countDepartment, SMALL_COL) . padLeft(($total['тугр'] / $total['стр']) / $countDepartment, MEDIUM_COL) . "\n";
            echo padRigth('Всего', LARGE_COL) . padLeft($total['сотр'], SMALL_COL) .
                     padLeft($total['тугр'], MEDIUM_COL) . padLeft($total['кофе'], SMALL_COL) .
                     padLeft($total['стр'], SMALL_COL) . padLeft($total['тугр'] / $total['стр'], MEDIUM_COL) . "\n"; 
        }
    }
    
    class Department
    {
        private $name; //Имя компании
        private $listOfEmployee; //Список сотрудников
        
        public function __construct($name = 'Noname Department')
        {
            $this->name = $name;
            $this->listOfEmployee = array();
        }
        
        //Получить имя департамента
        public function getDepartmentName()
        {
            return $this->name;
        }
        
        //Добавить сотрудника в департамент
        public function addEmployeeToDepartment($newEmployee)
        {
            $this->listOfEmployee[] = $newEmployee;
        }
        
        //Количество сотрудников
        public function getCountEmployee()
        {
            return count($this->listOfEmployee);
        }

        //Общая сумма
        public function getTotalSalary()
        {
            $totalSalary = 0;
            foreach ($this->listOfEmployee as $employee) {
                $totalSalary += $employee->getSalary();
            }
            return $totalSalary;
        }
        
        //Общий объем страниц
        public function getTotalPages()
        {
            $totalPages = 0;
            foreach ($this->listOfEmployee as $employee) {
                $totalPages += $employee->getPages();
            }
            return $totalPages;
        }
        
        //Обцее число кружек кофе
        public function getTotalCups()
        {
            $totalCups = 0;
            foreach ($this->listOfEmployee as $employee) {
                $totalCups += $employee->getCupOfCoffee();
            }
            return $totalCups;
        }
    }
    
    class Employee
    {
        protected $name; //Имя сотрудника
        protected  $isDirector; //Является ли он руководителем
        protected  $rank; //ранг
        protected  $proffession = ''; //Профессия
        protected  $salary = 0; //Зарплата
        protected  $pages = 0; //Страниц отчета
        protected  $cupOfCoffee = 0; //Кружек кофе
        
        public function __construct($name = 'Username', $rank = 0, $isDirector = false)
        {
            $this->name = $name;
            $this->rank = $rank;
            $this->isDirector = $isDirector;
        }

        //Получить имя сотрудника
        public function getEmployeeName()
        {
            return $this->name;
        }
        
        //Получить профессию сотрудника
        public function getEmployeeProffession()
        {
            return $this->proffession;
        }
        
        //Получить ранг сотрудника
        public function getEmployeeRank()
        {
            return $this->rank;
        }
        
        //Посчитать зарплату
        public function getSalary()
        {
            $totalSalary = $this->salary;
            if ($this->rank == 2) {
                $totalSalary *= 1.25;
            } elseif ($this->rank == 3) {
                $totalSalary *= 1.5;
            }
            
            if ($this->isDirector) {
                $totalSalary *= 1.5;
            }
            return $totalSalary;
        }
        
        //Посчитать количество страниц
        public function getPages()
        {
            return ($this->isDirector ? 0 : $this->pages);
        }
        
        //Посчитать количество кружек кофе
        public function getCupOfCoffee()
        {
            return ($this->isDirector ? ($this->cupOfCoffee * 2) : $this->cupOfCoffee);
        }
        
    }
    
    class Manager extends Employee
    {
        protected  $salary = 500;
        protected  $pages = 200;
        protected  $cupOfCoffee = 20;
        protected  $proffession = 'Менеджер';
    }
    
    class Marketer extends Employee
    {
        protected  $salary = 400;
        protected  $pages = 150;
        protected  $cupOfCoffee = 15;
        protected  $proffession = 'Маркетолог';
    }
    
    class Engineer extends Employee
    {
        protected  $salary = 200;
        protected  $pages = 50;
        protected  $cupOfCoffee = 5;
        protected  $proffession = 'Инженер';
    }
    
    class Analyst extends Employee
    {
        protected  $salary = 800;
        protected  $pages = 5;
        protected  $cupOfCoffee = 50;
        protected  $proffession = 'Аналитик';
    }
    
    //Для форматированного вывода
    function padLeft($text, $length)
    {
        if (mb_strlen($text) < $length) {
            $text = str_repeat(" ", $length - mb_strlen($text)) . $text;
        }
        
        return $text;
    }
    
    function padRigth($text, $length) {
        if (mb_strlen($text) < $length) {
            $text .= str_repeat(" ", $length - mb_strlen($text));
        }        
        
        return $text;
    }
    
    //функция принимает на вход строку вида: 9xме1;3xмe2;2xмe3;2xмa1;1xмe2 (последний идет руководитель) и возвращает массив созданных объектов
    function createEmployee($message)
    {
        $employee = array();
        
        $people = explode(';', $message);
        $totalPeople = count($people);
        
        
        for ($i = 0; $i < $totalPeople; $i++) {
            $rank = (int) mb_substr($people[$i], mb_strlen($people[$i]) - 1, 1);
            $proffession = mb_substr($people[$i], mb_strpos($people[$i], 'x') + 1, 2);
            $countEmployee = (int) $people[$i];
        
            $isDirector = ($i == ($totalPeople - 1)) ? true : false;
        
            for ($j = 0; $j < $countEmployee; $j++ ) {
                switch ($proffession) {
                    case "ме": $employee[] = new Manager("Имя{$j}", $rank, $isDirector);
                               break;
                    case "ма": $employee[] = new Marketer("Имя{$j}", $rank, $isDirector);
                               break;
                    case "ан": $employee[] = new Analyst("Имя{$j}", $rank, $isDirector);
                               break;
                    case "ин": $employee[] = new Engineer("Имя{$j}", $rank, $isDirector);
                               break;
                    default: echo "Проверьте входные данные! {$proffession}";
                             exit();
                }
            }
        }

        return $employee;        
    }
    
    //Добавляем каждого сотрудника в определенный департамент
    function addAllEmployeeToDepartment($employee, $departament)
    {
        $totalEmployee = count($employee);
        
        for ($i = 0; $i < $totalEmployee; $i++) {
            $departament->addEmployeeToDepartment($employee[$i]);
        }
    }
    
    //Создаем нашу компанию
    $vector = new Company('Вектор');

    //Создаем департаменты и добавляем их в команию
    $procurementDepartment = new Department('Департамент закупок');
    $vector->addDepartmentToCompany($procurementDepartment);
    
    $salesDepartament = new Department('Департамент продаж');
    $vector->addDepartmentToCompany($salesDepartament);
    
    $advertisingDepartment = new Department('Департамент рекламы');
    $vector->addDepartmentToCompany($advertisingDepartment);
    
    $logisticsDepartament = new Department('Департамент логистики');
    $vector->addDepartmentToCompany($logisticsDepartament);
    
    //Создаем сотрудников и добавляем их в департаменты
    $procurementEmployee = array();
    $procurementEmployee = createEmployee('9xме1;3xме2;2xме3;2xма1;1xме2');
    addAllEmployeeToDepartment($procurementEmployee, $procurementDepartment);
    
    $salesEmployee = array();
    $salesEmployee = createEmployee('12xме1;6xма1;3xан1;2xан2;1xма2');
    addAllEmployeeToDepartment($salesEmployee, $salesDepartament);
    
    $advertisingEmployee = array();
    $advertisingEmployee = createEmployee('15xма1;10xма2;8xме1;2xин1;1xма3');
    addAllEmployeeToDepartment($advertisingEmployee, $advertisingDepartment);
    
    $logisticsEmployee = array();
    $logisticsEmployee = createEmployee('13xме1;5xме2;5xин1;1xме1');
    addAllEmployeeToDepartment($logisticsEmployee, $logisticsDepartament);
    
    //Выводим отчет о компании
    $vector->printInfoAboutCompany();  