2007-07-19 10:40:28 +00:00
< ? php
2008-02-25 02:10:37 +00:00
/**
2008-08-06 02:43:46 +00:00
* Standard basic search form which conducts a fulltext search on all { @ link SiteTree }
* objects .
*
2008-09-26 04:35:29 +00:00
* @ see Use ModelController and SearchContext for a more generic search implementation based around DataObject
2008-02-25 02:10:37 +00:00
* @ package sapphire
* @ subpackage search
*/
2007-07-19 10:40:28 +00:00
class SearchForm extends Form {
2008-11-22 03:33:00 +00:00
/**
* @ var boolean $showInSearchTurnOn
* @ deprecated 2.3 SiteTree -> ShowInSearch should always be respected
*/
2007-07-19 10:40:28 +00:00
protected $showInSearchTurnOn ;
2008-11-22 03:33:00 +00:00
/**
* @ var int $numPerPage How many results are shown per page .
* Relies on pagination being implemented in the search results template .
*/
protected $numPerPage = 10 ;
/**
*
* @ param Controller $controller
* @ param string $name The name of the form ( used in URL addressing )
* @ param FieldSet $fields Optional , defaults to a single field named " Search " . Search logic needs to be customized
* if fields are added to the form .
* @ param FieldSet $actions Optional , defaults to a single field named " Go " .
* @ param boolean $showInSearchTurnOn DEPRECATED 2.3
*/
2007-07-19 10:40:28 +00:00
function __construct ( $controller , $name , $fields = null , $actions = null , $showInSearchTurnOn = true ) {
$this -> showInSearchTurnOn = $showInSearchTurnOn ;
2008-11-11 21:16:51 +00:00
2007-07-19 10:40:28 +00:00
if ( ! $fields ) {
2008-11-11 21:16:51 +00:00
$fields = new FieldSet (
new TextField ( 'Search' , _t ( 'SearchForm.SEARCH' , 'Search' )
));
2007-07-19 10:40:28 +00:00
}
2008-11-11 21:16:51 +00:00
2007-07-19 10:40:28 +00:00
if ( ! $actions ) {
$actions = new FieldSet (
2008-02-25 02:10:37 +00:00
new FormAction ( " getResults " , _t ( 'SearchForm.GO' , 'Go' ))
2007-07-19 10:40:28 +00:00
);
}
// We need this because it's a get form. It can't go in the action value
// Hayden: Sorry if I've got it mixed up, but on the results or not found pages, the
// RelativeLink seems to be empty and it packs a sad
$formController = isset ( $_GET [ 'formController' ]) ? $_GET [ 'formController' ] : null ;
2008-11-11 21:16:51 +00:00
if ( ! $formController ) $formController = $controller -> RelativeLink ();
$fields -> push ( new HiddenField ( 'formController' , null , $formController ));
$fields -> push ( new HiddenField ( 'executeForm' , null , $name ));
2007-07-19 10:40:28 +00:00
parent :: __construct ( $controller , $name , $fields , $actions );
2007-10-28 21:44:38 +00:00
$this -> disableSecurityToken ();
2007-07-19 10:40:28 +00:00
}
function FormMethod () {
return " get " ;
}
public function forTemplate (){
2008-11-11 21:16:51 +00:00
return $this -> renderWith ( array (
'SearchForm' ,
'Form'
));
2007-07-19 10:40:28 +00:00
}
/**
* Return dataObjectSet of the results using $_REQUEST to get info from form .
2008-11-22 03:33:00 +00:00
* Wraps around { @ link searchEngine ()} .
*
* @ param int $numPerPage DEPRECATED 2.3 Use SearchForm -> numPerPage
* @ param array $data Request data as an associative array . Should contain at least a key 'Search' with all searched keywords .
* @ return DataObjectSet
2007-07-19 10:40:28 +00:00
*/
2008-11-22 03:33:00 +00:00
public function getResults ( $numPerPage = null , $data = null ){
// legacy usage: $data was defaulting to $_REQUEST, parameter not passed in doc.silverstripe.com tutorials
if ( ! isset ( $data )) $data = $_REQUEST ;
$keywords = $data [ 'Search' ];
2007-07-19 10:40:28 +00:00
$andProcessor = create_function ( '$matches' , '
return " + " . $matches [ 2 ] . " + " . $matches [ 4 ] . " " ;
' );
$notProcessor = create_function ( '$matches' , '
return " - " . $matches [ 3 ];
' );
$keywords = preg_replace_callback ( '/()("[^()"]+")( and )("[^"()]+")()/i' , $andProcessor , $keywords );
$keywords = preg_replace_callback ( '/(^| )([^() ]+)( and )([^ ()]+)( |$)/i' , $andProcessor , $keywords );
$keywords = preg_replace_callback ( '/(^| )(not )("[^"()]+")/i' , $notProcessor , $keywords );
$keywords = preg_replace_callback ( '/(^| )(not )([^() ]+)( |$)/i' , $notProcessor , $keywords );
$keywords = $this -> addStarsToKeywords ( $keywords );
if ( strpos ( $keywords , '"' ) !== false || strpos ( $keywords , '+' ) !== false || strpos ( $keywords , '-' ) !== false || strpos ( $keywords , '*' ) !== false ) {
2008-11-22 03:33:00 +00:00
$results = $this -> searchEngine ( $keywords , $numPerPage , " Relevance DESC " , " " , true );
2007-07-19 10:40:28 +00:00
} else {
2008-11-22 03:33:00 +00:00
$results = $this -> searchEngine ( $keywords , $numPerPage );
}
// filter by permission
if ( $results ) foreach ( $results as $result ) {
if ( ! $result -> canView ()) $results -> remove ( $result );
}
return $results ;
2007-07-19 10:40:28 +00:00
}
2008-11-22 03:33:00 +00:00
protected function addStarsToKeywords ( $keywords ) {
2007-07-19 10:40:28 +00:00
if ( ! trim ( $keywords )) return " " ;
// Add * to each keyword
$splitWords = split ( " + " , trim ( $keywords ));
while ( list ( $i , $word ) = each ( $splitWords )) {
if ( $word [ 0 ] == '"' ) {
while ( list ( $i , $subword ) = each ( $splitWords )) {
$word .= ' ' . $subword ;
if ( substr ( $subword , - 1 ) == '"' ) break ;
}
} else {
$word .= '*' ;
}
$newWords [] = $word ;
}
return implode ( " " , $newWords );
}
2008-11-22 03:33:00 +00:00
2007-07-19 10:40:28 +00:00
/**
* The core search engine , used by this class and its subclasses to do fun stuff .
* Searches both SiteTree and File .
2008-11-22 03:33:00 +00:00
*
* @ param string $keywords Keywords as a string .
2007-07-19 10:40:28 +00:00
*/
2008-11-22 03:33:00 +00:00
public function searchEngine ( $keywords , $numPerPage = null , $sortBy = " Relevance DESC " , $extraFilter = " " , $booleanSearch = false , $alternativeFileFilter = " " , $invertedMatch = false ) {
if ( ! $numPerPage ) $numPerPage = $this -> numPerPage ;
2007-07-19 10:40:28 +00:00
$fileFilter = '' ;
$keywords = addslashes ( $keywords );
if ( $booleanSearch ) $boolean = " IN BOOLEAN MODE " ;
if ( $extraFilter ) {
$extraFilter = " AND $extraFilter " ;
if ( $alternativeFileFilter ) $fileFilter = " AND $alternativeFileFilter " ;
else $fileFilter = $extraFilter ;
}
if ( $this -> showInSearchTurnOn ) $extraFilter .= " AND showInSearch <> 0 " ;
2007-10-17 03:22:37 +00:00
$start = isset ( $_GET [ 'start' ]) ? ( int ) $_GET [ 'start' ] : 0 ;
$limit = $start . " , " . ( int ) $numPerPage ;
2007-07-19 10:40:28 +00:00
$notMatch = $invertedMatch ? " NOT " : " " ;
if ( $keywords ) {
$matchContent = " MATCH (Title, MenuTitle, Content, MetaTitle, MetaDescription, MetaKeywords) AGAINST (' $keywords ' $boolean ) " ;
$matchFile = " MATCH (Filename, Title, Content) AGAINST (' $keywords ' $boolean ) AND ClassName = 'File' " ;
// We make the relevance search by converting a boolean mode search into a normal one
$relevanceKeywords = str_replace ( array ( '*' , '+' , '-' ), '' , $keywords );
$relevanceContent = " MATCH (Title) AGAINST (' $relevanceKeywords ') + MATCH (Title, MenuTitle, Content, MetaTitle, MetaDescription, MetaKeywords) AGAINST (' $relevanceKeywords ') " ;
$relevanceFile = " MATCH (Filename, Title, Content) AGAINST (' $relevanceKeywords ') " ;
} else {
$relevanceContent = $relevanceFile = 1 ;
$matchContent = $matchFile = " 1 = 1 " ;
}
$queryContent = singleton ( 'SiteTree' ) -> extendedSQL ( $notMatch . $matchContent . $extraFilter , " " );
$baseClass = reset ( $queryContent -> from );
// There's no need to do all that joining
2008-11-23 00:31:06 +00:00
$queryContent -> from = array ( str_replace ( array ( '`' , '"' ), '' , $baseClass ) => $baseClass );
2008-11-24 19:28:46 +00:00
$queryContent -> select = array ( " \" ClassName \" " , " $baseClass . \" ID \" " , " \" ParentID \" " , " \" Title \" " ,
" \" URLSegment \" " , " \" Content \" " , " \" LastEdited \" " , " \" Created \" " , " '' AS \" Filename \" " ,
" '' AS \" Name \" " , " $relevanceContent AS \" Relevance \" " , " \" CanViewType \" " );
2007-07-19 10:40:28 +00:00
$queryContent -> orderby = null ;
$queryFiles = singleton ( 'File' ) -> extendedSQL ( $notMatch . $matchFile . $fileFilter , " " );
$baseClass = reset ( $queryFiles -> from );
// There's no need to do all that joining
2008-11-23 00:31:06 +00:00
$queryFiles -> from = array ( str_replace ( array ( '`' , '"' ), '' , $baseClass ) => $baseClass );
2008-11-24 19:28:46 +00:00
$queryFiles -> select = array ( " \" ClassName \" " , " $baseClass . \" ID \" " , " '' AS \" ParentID \" " , " \" Title \" " ,
" '' AS \" URLSegment \" " , " \" Content \" " , " \" LastEdited \" " , " \" Created \" " , " \" Filename \" " , " \" Name \" " ,
2008-11-24 09:31:14 +00:00
" $relevanceFile AS \" Relevance \" " , " NULL AS \" CanViewType \" " );
2007-07-19 10:40:28 +00:00
$queryFiles -> orderby = null ;
$fullQuery = $queryContent -> sql () . " UNION " . $queryFiles -> sql () . " ORDER BY $sortBy LIMIT $limit " ;
$totalCount = $queryContent -> unlimitedRowCount () + $queryFiles -> unlimitedRowCount ();
$records = DB :: query ( $fullQuery );
foreach ( $records as $record )
$objects [] = new $record [ 'ClassName' ]( $record );
2007-10-17 03:26:25 +00:00
if ( isset ( $objects )) $doSet = new DataObjectSet ( $objects );
else $doSet = new DataObjectSet ();
2007-07-19 10:40:28 +00:00
2008-02-25 02:10:37 +00:00
$doSet -> setPageLimits ( $start , $numPerPage , $totalCount );
2007-07-19 10:40:28 +00:00
return $doSet ;
}
2008-11-11 21:16:51 +00:00
2008-11-22 03:33:00 +00:00
/**
* Get the search query for display in a " You searched for ... " sentence .
*
* @ param array $data
* @ return string
*/
public function getSearchQuery ( $data = null ) {
// legacy usage: $data was defaulting to $_REQUEST, parameter not passed in doc.silverstripe.com tutorials
if ( ! isset ( $data )) $data = $_REQUEST ;
return Convert :: raw2xml ( $data [ 'Search' ]);
2007-07-19 10:40:28 +00:00
}
}
2008-08-06 02:43:46 +00:00
?>