<?php
    //Убрал в функция createEmployee, padLeft, padRight - тайп-хинт
    //Для int и string, т.к. ideone считает их за объекты
    
    error_reporting(-1);
    mb_internal_encoding('utf-8');
  
    define("SMALL_COL", 7);
    define("MEDIUM_COL", 12);
    define("LARGE_COL", 20);
  
    class Company
    {
        private $name; 
        private $listOfDepatments;
        
        public function __construct($name)
        {
            $this->name = $name;
            $this->listOfDepatments = array();
        }

        /**
         *  Метод возвращающий имя компании
         *
         *  @return string  Имя компании
         */
        public function getNameCompany()
        {
            return $this->name;
        }
        
        /**
         *  Метод возвращающий количество департаментов
         *
         *  @return integer  Имя компании
         */
        public function getCountDeparments()
        {
            return count($this->listOfDepatments);
        }
        
        /**
         *  Метод добавляющий новый департамент в компанию
         *
         *  @param Department $newDepartment Добавляемый департамент
         *  @return void
         */
        public function addDepartmentToCompany(Department $newDepartment)
        {
            $this->listOfDepatments[] = $newDepartment;
        }
        
        /**
         *  Метод возвращающий список департаментов
         *
         *  @return array  Массив департаментов
         */
        public function getListOfDepartments()
        {
            return $this->listOfDepatments;
        }
    }
    
    class Department
    {
        private $name;
        private $listOfEmployee;
        
        public function __construct($name)
        {
            $this->name = $name;
            $this->listOfEmployee = array();
        }
        
        /**
         *  Метод возвращающий имя департамента
         *
         *  @return string  Имя департамент
         */
        public function getDepartmentName()
        {
            return $this->name;
        }
        
        /**
         *  Метод добавляющий нового сотрудника
         *
         *  $param Employee $newEmployee Новый сотрудник
         *  @return void
         */
        public function addEmployeeToDepartment(Employee $newEmployee)
        {
            $this->listOfEmployee[] = $newEmployee;
        }
        
        /**
         *  Метод возвращающий количество сотрудников
         *
         *  @return integer  Количество сотрудников
         */
        public function getCountEmployee()
        {
            return count($this->listOfEmployee);
        }

        /**
         *  Метод возвращающий зарплату всего департамента
         *
         *  @return integer  Зарплата департамента
         */
        public function getTotalSalary()
        {
            $totalSalary = 0;
            foreach ($this->listOfEmployee as $employee) {
                $totalSalary += $employee->getSalary();
            }
            return $totalSalary;
        }
        
        /**
         *  Метод возвращающий общее число наработанных страниц
         *
         *  @return integer  Количество страниц
         */
        public function getTotalPages()
        {
            $totalPages = 0;
            foreach ($this->listOfEmployee as $employee) {
                $totalPages += $employee->getPages();
            }
            return $totalPages;
        }
        
        /**
         *  Метод возвращающий общее число выпитых кружен
         *
         *  @return integer  Число выпитых кружек
         */
        public function getTotalCups()
        {
            $totalCups = 0;
            foreach ($this->listOfEmployee as $employee) {
                $totalCups += $employee->getCupOfCoffee();
            }
            return $totalCups;
        }
    }
    
    abstract class Employee
    {
        protected  $isDirector;
        protected  $rank;
        protected  $proffession;
        protected  $salary;
        protected  $pages;
        protected  $cupOfCoffee;
        
        public function __construct($rank, $isDirector = false)
        {
            $this->rank = $rank;
            $this->isDirector = $isDirector;
            $this->setSalary();
            $this->setPages();
            $this->setCupOfCoffee();
            $this->setProffession();
        }
        
        abstract protected function setSalary();
        abstract protected function setPages();
        abstract protected function setCupOfCoffee();
        abstract protected function setProffession();
        
        /**
         *  Метод возвращающий профессию сотрудника
         *
         *  @return string Профессия сотрудника
         */
        public function getEmployeeProffession()
        {
            return $this->proffession;
        }
        
        /**
         *  Метод возвращающий ранг сотрудника
         *
         *  @return integer Ранг сотрудника
         */
        public function getEmployeeRank()
        {
            return $this->rank;
        }
        
        /**
         *  Метод возвращающий зарплату сотрудника
         *
         *  @return integer  Зарплата сотрудника
         */
        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;
        }
        
        /**
         *  Метод возвращающий количество наработанных страниц сотрудником
         *
         *  @return integer Общее число страниц сотрудника
         */
        public function getPages()
        {
            return ($this->isDirector ? 0 : $this->pages);
        }
        
        /**
         *  Метод возвращающий количество выпитых кружек сотрудником
         *
         *  @return integer Количество выпитых кружек
         */
        public function getCupOfCoffee()
        {
            return ($this->isDirector ? ($this->cupOfCoffee * 2) : $this->cupOfCoffee);
        }
        
    }
    
    class Manager extends Employee
    {
        protected function setSalary()
        {
            $this->salary = 500;
        }
        
        protected function setPages()
        {
            $this->pages = 200;
        }
        
        protected function setCupOfCoffee()
        {
            $this->cupOfCoffee = 20;
        }

        protected  function setProffession()
        {
            $this->proffession = 'Менеджер';
        }
    }
    
    class Marketer extends Employee
    {
        protected function setSalary()
        {
            $this->salary = 400;
        }
        
        protected function setPages()
        {
            $this->pages = 150;
        }
        
        protected function setCupOfCoffee()
        {
            $this->cupOfCoffee = 15;
        }

        protected  function setProffession()
        {
            $this->proffession = 'Маркетолог';
        }
    }
    
    class Engineer extends Employee
    {
        protected function setSalary()
        {
            $this->salary = 200;
        }
        
        protected function setPages()
        {
            $this->pages = 50;
        }
        
        protected function setCupOfCoffee()
        {
            $this->cupOfCoffee = 5;
        }

        protected  function setProffession()
        {
            $this->proffession = 'Инженер';
        }
    }
    
    class Analyst extends Employee
    {
        protected function setSalary()
        {
            $this->salary = 800;
        }
        
        protected function setPages()
        {
            $this->pages = 5;
        }
        
        protected function setCupOfCoffee()
        {
            $this->cupOfCoffee = 50;
        }

        protected  function setProffession()
        {
            $this->proffession = 'Аналитик';
        }
    }
    
    /**
     *  Класс для создания отчета по компании
     */
    class CompanyReport
    {
        public function printInfoAboutCompany(Company $company)
        {
            $countDepartment = $company->getCountDeparments();
            $total = array('сотр' => 0, 'тугр' => 0, 'кофе' => 0, 'стр' => 0);

            echo "\"{$company->getNameCompany()}\"\n";
            echo padRigth("Деп. ", LARGE_COL) . padLeft("сотр.", SMALL_COL) . padLeft("тугр.", MEDIUM_COL) .
                 padLeft("кофе", SMALL_COL) . padLeft("стр.", SMALL_COL) . padLeft("тугр./стр.", LARGE_COL) . "\n";
            $listOfDepatments = $company->getListOfDepartments();
            foreach ($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(), MEDIUM_COL) . padLeft($departament->getTotalSalary() / $departament->getTotalPages(), LARGE_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, MEDIUM_COL) . padLeft(($total['тугр'] / $total['стр']) / $countDepartment, LARGE_COL) . "\n";
            echo padRigth('Всего', LARGE_COL) . padLeft($total['сотр'], SMALL_COL) .
                     padLeft($total['тугр'], MEDIUM_COL) . padLeft($total['кофе'], SMALL_COL) .
                     padLeft($total['стр'], MEDIUM_COL) . padLeft($total['тугр'] / $total['стр'], LARGE_COL) . "\n"; 
        }
    }
    
    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;
    }
    
    /**
     *  Функция создания массива сотрудников
     *
     *  @param string $message Строка вида:9xме1;3xмe2;2xмe3;2xмa1;1xмe2 (последний идет руководитель) 
     *  @return array Список сотрудников
     */
    function createEmployee($message)
    {
        $employee = array();
        
        $people = explode(';', $message);
        $totalPeople = count($people);
        
        
        for ($i = 0; $i < $totalPeople; $i++) {
            
            $matches = array();
            preg_match('/([0-9]+)x([а-яё]+)([0-9])/iu', $people[$i], $matches);
             
            $countEmployee = (int) $matches[1];
            $proffession = $matches[2];
            $rank = (int) $matches[3];
        
            $isDirector = ($i == ($totalPeople - 1)) ? true : false;
        
            for ($j = 0; $j < $countEmployee; $j++ ) {
                switch ($proffession) {
                    case "ме": $employee[] = new Manager($rank, $isDirector);
                               break;
                    case "ма": $employee[] = new Marketer($rank, $isDirector);
                               break;
                    case "ан": $employee[] = new Analyst($rank, $isDirector);
                               break;
                    case "ин": $employee[] = new Engineer($rank, $isDirector);
                               break;
                    default: echo "Проверьте входные данные! {$proffession}";
                             exit();
                }
            }
        }

        return $employee;        
    }
    
    /**
     *  Функция добавляет созданный список сотрудников в департамент
     *
     *  @return void
     */
    function addAllEmployeeToDepartment(array $employee, Department $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);
    

    $newReport = new CompanyReport();
    $newReport->printInfoAboutCompany($vector);  