<?php
/*
Программа: поиск пути
Версия:5.0
На данном этапе программа:
-ищет самый короткий путь из точки A в B
-показывает время достижения пути
-показывает способ достижения пути
-показывает, на чём добираться
-сколько времени это займёт
Todo:
-возможно, сделать для метро
-изменить структуру кода (что-бы путь Медведково=>Проспект Мира не показывал все промежуточные станции)
-долго вбивать все станции (взять в Яндексе или найти где-нибудь еще)
-можно сделать 2-3 ветки и не париться
-сделать web-интерфейс с приложенной картой (http://content-available-to-author-only.cc/55nguY) и формой
-немного современного дизайна
*/
SUBWAY => 'на метро' ,
FOOT => 'пешком' ,
BUS => 'на автобусе'
) ;
'pet' => 'ст. м. Петроградская' ,
'chk' => 'ст. м. Чкаловская' ,
'gor' => 'ст. м. Горьковская' ,
'spo' => 'ст. м. Спортивная' ,
'vas' => 'ст. м. Василеостровская' ,
'kre' => 'Петропавловская крепость' ,
'let' => 'Летний сад' ,
'dvo' => 'Дворцовая площадь' ,
'isa' => 'Исакиевский собор' ,
'nov' => 'Новая Голландия' ,
'ras' => 'Дом Раскольникова' ,
'gos' => 'Гостиный Двор' ,
'sen' => 'Сенная Площадь' ,
'vla' => 'ст. м. Владимирская' ,
'vit' => 'Витебский вокзал' ,
'teh' => 'Технологический Институт'
) ;
'chk' => canGet( 10 , BUS) ,
'gor' => canGet( 3 , SUBWAY)
) ,
'pet' => canGet( 10 , BUS) ,
'spo' => canGet( 3 , SUBWAY)
) ,
'pet' => canGet( 3 , BUS) ,
'kre' => canGet( 5 , FOOT) ,
'gos' => canGet( 6 , SUBWAY)
) ,
'chk' => canGet( 3 , SUBWAY) ,
'vas' => canGet( 10 , BUS) ,
'sen' => canGet( 7 , SUBWAY)
) ,
'spo' => canGet( 10 , BUS) ,
'gos' => canGet( 7 , SUBWAY) ,
'nov' => canGet( 11 , FOOT)
) ,
'gor' => canGet( 5 , FOOT)
) ,
'dvo' => canGet( 6 , FOOT) ,
'gos' => canGet( 7 , FOOT)
) ,
'isa' => canGet( 6 , FOOT) ,
'gos' => canGet( 6 , FOOT) ,
'let' => canGet( 6 , FOOT)
) ,
'dvo' => canGet( 6 , FOOT) ,
'nov' => canGet( 5 , FOOT)
) ,
'vas' => canGet( 11 , FOOT) ,
'isa' => canGet( 5 , FOOT) ,
'ras' => canGet( 7 , BUS)
) ,
'nov' => canGet( 7 , BUS) ,
'sen' => canGet( 3 , FOOT)
) ,
'vas' => canGet( 7 , SUBWAY) ,
'sen' => canGet( 3 , SUBWAY) ,
'dvo' => canGet( 6 , FOOT) ,
'gor' => canGet( 6 , SUBWAY) ,
'let' => canGet( 7 , FOOT) ,
'vla' => canGet( 7 , FOOT)
) ,
'ras' => canGet( 3 , FOOT) ,
'spo' => canGet( 7 , SUBWAY) ,
'gos' => canGet( 3 , SUBWAY) ,
'vla' => canGet( 4 , SUBWAY) ,
'vit' => canGet( 2 , SUBWAY) ,
'teh' => canGet( 3 , SUBWAY)
) ,
'sen' => canGet( 4 , SUBWAY) ,
'gos' => canGet( 7 , FOOT) ,
'vit' => canGet( 3 , SUBWAY)
) ,
'sen' => canGet( 2 , SUBWAY) ,
'teh' => canGet( 2 , SUBWAY) ,
'vla' => canGet( 3 , SUBWAY)
) ,
'sen' => canGet( 3 , SUBWAY) ,
'vit' => canGet( 2 , SUBWAY)
)
) ;
/* Чтобы не писать много раз array('time' => ..., 'by' => ...), используем функцию.
«canGet» переводится как «можно попасть» */
function canGet( $time , $byWhat ) {
return array ( 'time' => $time , 'by' => $byWhat ) ; }
function find_way( $paths , $from , $where , $time , $way ) {
$way [ ] = $from ;
}
$current = $way [ count ( $way ) - 1 ] ; //на какой мы сейчас станции
//Считаем время
$prev = $way [ count ( $way ) - 2 ] ; //Предыдущая станция(для времени) $time += $paths [ $prev ] [ $current ] [ 'time' ] ;
}
$way [ ] = $where ;
$time += $paths [ $current ] [ $where ] [ 'time' ] ; //Время текущей с следующей (,которая цель)
$arrPrep [ 'way' ] = $way ;
$arrPrep [ 'time' ] = $time ;
$GLOBALS [ 'ways' ] [ ] = $arrPrep ; //Добавляем вариант прохода в возвращаемый функцией массив
}
else {
foreach ( $paths [ $current ] as $k => $v ) {
$next = $k ;
$way [ ] = $next ;
find_way( $paths , $from , $where , $time , $way ) ;
}
}
}
if ( isset ( $GLOBALS [ 'ways' ] ) ) return ( $GLOBALS [ 'ways' ] ) ;
}
////////////////////////////////////
function find_shortest_way( $paths , $from , $where , $time , $way ) {
$ways = find_way( $paths , $from , $where , $time , $way ) ;
$shortest [ 'time' ] = 0 ;
$shortest [ 'way' ] = '' ;
foreach ( $ways as $k => $v ) {
if ( ( $shortest [ 'time' ] > $ways [ $k ] [ 'time' ] ) || ( $shortest [ 'time' ] == 0 ) ) {
$shortest [ 'time' ] = $ways [ $k ] [ 'time' ] ;
$shortest [ 'way' ] = $ways [ $k ] [ 'way' ] ;
}
}
return $shortest ;
}
////////////////////////////////////
function show_shortest_way( $paths , $from , $where , $time , $way , $pointNames , $transportName ) {
$ways = find_shortest_way( $paths , $from , $where , $time , $way ) ;
$first = $pointNames [ reset ( $ways [ 'way' ] ) ] ; $last = $pointNames [ end ( $ways [ 'way' ] ) ] ;
echo "Как доехать из " . $first . " в " . $last . " за " . $ways [ 'time' ] . " минут\n " ;
for ( $i = 0 ; $i < count ( $ways [ 'way' ] ) ; $i ++ ) { $current = $ways [ 'way' ] [ $i ] ;
if ( $i < ( count ( $ways [ 'way' ] ) - 1 ) ) { $next = $ways [ 'way' ] [ $i + 1 ] ;
echo "" . $pointNames [ $current ] . "\n =>" ;
echo $transportName [ $paths [ $current ] [ $next ] [ 'by' ] ] . "" ;
echo " за " . $paths [ $current ] [ $next ] [ 'time' ] . " минут\n " ;
}
else
echo $pointNames [ $current ] . "\n \n " ;
}
}
////////////////////////////////////
show_shortest_way( $paths , 'pet' , 'teh' , 0 , null , $pointNames , $transportName ) ;
show_shortest_way( $paths , 'vas' , 'let' , 0 , null , $pointNames , $transportName ) ;
<?php

/*
Программа: поиск пути
Версия:5.0
На данном этапе программа:
	-ищет самый короткий путь из точки A в B
	-показывает время достижения пути
	-показывает способ достижения пути
		-показывает, на чём добираться
		-сколько времени это займёт
		
		
Todo:
-возможно, сделать для метро
	-изменить структуру кода (что-бы путь Медведково=>Проспект Мира не показывал все промежуточные станции)
	-долго вбивать все станции (взять в Яндексе или найти где-нибудь еще)
	-можно сделать 2-3 ветки и не париться
	
-сделать web-интерфейс с приложенной картой (http://content-available-to-author-only.cc/55nguY) и формой
-немного современного дизайна
	
*/
error_reporting(-1);

define('SUBWAY', 'sub');
define('FOOT', 'foot');
define('BUS', 'bus');

$transportName = array(
    SUBWAY  =>  'на метро',
    FOOT    =>  'пешком',
    BUS     =>  'на автобусе'
);

$pointNames = array(
    'pet'   =>  'ст. м. Петроградская',
    'chk'   =>  'ст. м. Чкаловская',
    'gor'   =>  'ст. м. Горьковская',
    'spo'   =>  'ст. м. Спортивная',
    'vas'   =>  'ст. м. Василеостровская',
    'kre'   =>  'Петропавловская крепость',
    'let'   =>  'Летний сад',
    'dvo'   =>  'Дворцовая площадь',
    'isa'   =>  'Исакиевский собор',
    'nov'   =>  'Новая Голландия',
    'ras'   =>  'Дом Раскольникова',
    'gos'   =>  'Гостиный Двор',
    'sen'   =>  'Сенная Площадь',
    'vla'   =>  'ст. м. Владимирская',
    'vit'   =>  'Витебский вокзал',
    'teh'   =>  'Технологический Институт'
);

$paths = array(
    'pet'   =>  array(
        'chk'   =>  canGet(10, BUS),
        'gor'   =>  canGet(3, SUBWAY)
    ),

    'chk'   =>  array(
        'pet'   =>  canGet(10, BUS),
        'spo'   =>  canGet(3, SUBWAY)
    ),

    'gor'   =>  array(
        'pet'   =>  canGet(3, BUS),
        'kre'   =>  canGet(5, FOOT),
        'gos'   =>  canGet(6, SUBWAY)
    ),

    'spo'   =>  array(
        'chk'   =>  canGet(3, SUBWAY),
        'vas'   =>  canGet(10, BUS),
        'sen'   =>  canGet(7, SUBWAY)
    ),

    'vas'   =>  array(
        'spo'   =>  canGet(10, BUS),
        'gos'   =>  canGet(7, SUBWAY),
        'nov'   =>  canGet(11, FOOT)
    ),

    'kre'   =>  array(
        'gor'   =>  canGet(5, FOOT)
    ),

    'let'   =>  array(
        'dvo'   =>  canGet(6, FOOT),
        'gos'   =>  canGet(7, FOOT)
    ),

    'dvo'   =>  array(
        'isa'   =>  canGet(6, FOOT),
        'gos'   =>  canGet(6, FOOT),
        'let'   =>  canGet(6, FOOT)
    ),

    'isa'   =>  array(
        'dvo'   =>  canGet(6, FOOT),
        'nov'   =>  canGet(5, FOOT)
    ),

    'nov'   =>  array(
        'vas'   =>  canGet(11, FOOT),
        'isa'   =>  canGet(5, FOOT),
        'ras'   =>  canGet(7, BUS)
    ),

    'ras'   =>  array(
        'nov'   =>  canGet(7, BUS),
        'sen'   =>  canGet(3, FOOT)
    ),

    'gos'   =>  array(
        'vas'   =>  canGet(7, SUBWAY),
        'sen'   =>  canGet(3, SUBWAY),
        'dvo'   =>  canGet(6, FOOT),
        'gor'   =>  canGet(6, SUBWAY),
        'let'   =>  canGet(7, FOOT),
        'vla'   =>  canGet(7, FOOT)        
    ),

    'sen'   =>  array(
        'ras'   =>  canGet(3, FOOT),
        'spo'   =>  canGet(7, SUBWAY),
        'gos'   =>  canGet(3, SUBWAY),
        'vla'   =>  canGet(4, SUBWAY),
        'vit'   =>  canGet(2, SUBWAY),
        'teh'   =>  canGet(3, SUBWAY)
    ),

    'vla'   =>  array(
        'sen'   =>  canGet(4, SUBWAY),
        'gos'   =>  canGet(7, FOOT),
        'vit'   =>  canGet(3, SUBWAY)
    ),

    'vit'   =>  array(
        'sen'   =>  canGet(2, SUBWAY),
        'teh'   =>  canGet(2, SUBWAY),
        'vla'   =>  canGet(3, SUBWAY)
    ),

    'teh'   =>  array(
        'sen'   =>  canGet(3, SUBWAY),
        'vit'   =>  canGet(2, SUBWAY)        
    )
);

/* Чтобы не писать много раз array('time' => ..., 'by' => ...), используем функцию. 
    «canGet» переводится как «можно попасть» */
function canGet($time, $byWhat) {
    return array('time'     =>  $time, 'by' =>  $byWhat);
}

function find_way($paths,$from,$where,$time,$way){

	if (!is_array($way)){
		$way=array();
		$way[]=$from;
	}

	$current=$way[count($way)-1]; //на какой мы сейчас станции
	
	//Считаем время
	if (count($way)>1){
		$prev=$way[count($way)-2]; //Предыдущая станция(для времени)
		$time+=$paths[$prev][$current]['time'];
	}

	
	if(array_key_exists($where,$paths[$current])){
		$way[]=$where;
		$time+=$paths[$current][$where]['time']; //Время текущей с следующей (,которая цель)

		$arrPrep['way']=$way;
		$arrPrep['time']=$time;
		$GLOBALS['ways'][]=$arrPrep;//Добавляем вариант прохода в возвращаемый функцией массив
	}
	else{
		foreach ($paths[$current] as $k=>$v){
			$next=$k;
			if(!in_array($next,$way)){
				$way[]=$next;
				find_way($paths,$from,$where,$time,$way);
				array_pop($way);
			}

		}
	}
	if (isset($GLOBALS['ways']))
		return ($GLOBALS['ways']);
}

////////////////////////////////////
function find_shortest_way($paths,$from,$where,$time,$way){
	$ways=find_way($paths,$from,$where,$time,$way);

	$shortest['time']=0;
	$shortest['way']='';
	
	foreach($ways as $k=>$v){
		if (($shortest['time']>$ways[$k]['time'])||($shortest['time']==0)){
			$shortest['time']=$ways[$k]['time'];
			$shortest['way']=$ways[$k]['way'];
		}		
	}

	return $shortest;
}
////////////////////////////////////
function show_shortest_way($paths,$from,$where,$time,$way,$pointNames,$transportName){
	$ways=find_shortest_way($paths,$from,$where,$time,$way);
	
	$first=$pointNames[reset($ways['way'])];
	$last=$pointNames[end($ways['way'])];
	
	echo "Как доехать из ".$first." в ".$last." за ".$ways['time']." минут\n";
	
	for($i=0;$i<count($ways['way']);$i++){
		$current=$ways['way'][$i];
		
		
		if ($i<(count($ways['way'])-1)){
			$next=$ways['way'][$i+1];
			echo "".$pointNames[$current]."\n	=>";
			echo $transportName[$paths[$current][$next]['by']]. "";
			echo " за ".$paths[$current][$next]['time']." минут\n";
		}
		else
			echo $pointNames[$current]."\n\n";
	}
}
////////////////////////////////////
show_shortest_way($paths,'pet','teh',0,null,$pointNames,$transportName);
show_shortest_way($paths,'vas','let',0,null,$pointNames,$transportName);