2010-10-04 04:32:48 +00:00
|
|
|
<?php
|
|
|
|
/**
|
2010-10-19 04:53:36 +00:00
|
|
|
* Provides a simple search engine for your site based on the MySQL FULLTEXT index.
|
|
|
|
* Adds the {@link FulltextSearchable} extension to data classes,
|
2011-03-29 17:55:13 +13:00
|
|
|
* as well as the {@link ContentControllerSearchExtension} to {@link ContentController}
|
|
|
|
* (if the 'cms' module is available as well).
|
2010-10-19 04:53:36 +00:00
|
|
|
* (this means you can use $SearchForm in your template without changing your own implementation).
|
|
|
|
*
|
2011-12-03 12:00:56 +01:00
|
|
|
* CAUTION: Will make all files in your /assets folder searchable by file name
|
|
|
|
* unless "File" is excluded from FulltextSearchable::enable().
|
|
|
|
*
|
2010-10-19 04:53:36 +00:00
|
|
|
* @see http://doc.silverstripe.org/tutorial:4-site-search
|
2010-10-19 03:54:51 +00:00
|
|
|
*
|
2010-10-13 03:53:12 +00:00
|
|
|
* @package sapphire
|
|
|
|
* @subpackage search
|
2010-10-04 04:32:48 +00:00
|
|
|
*/
|
2011-04-15 19:35:30 +10:00
|
|
|
class FulltextSearchable extends DataExtension {
|
2010-10-19 04:53:36 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @var String Comma-separated list of database column names
|
|
|
|
* that can be searched on. Used for generation of the database index defintions.
|
|
|
|
*/
|
2010-10-04 04:32:48 +00:00
|
|
|
protected $searchFields;
|
2010-10-19 04:53:36 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @var Array List of class names
|
|
|
|
*/
|
|
|
|
protected static $searchable_classes;
|
2010-10-19 03:54:51 +00:00
|
|
|
|
2010-10-04 04:32:48 +00:00
|
|
|
/**
|
|
|
|
* Enable the default configuration of MySQL full-text searching on the given data classes.
|
2010-10-19 04:53:36 +00:00
|
|
|
* It can be used to limit the searched classes, but not to add your own classes.
|
|
|
|
* For this purpose, please use {@link Object::add_extension()} directly:
|
|
|
|
* <code>
|
|
|
|
* Object::add_extension('MyObject', "FulltextSearchable('MySearchableField,'MyOtherField')");
|
|
|
|
* </code>
|
|
|
|
*
|
|
|
|
* Caution: This is a wrapper method that should only be used in _config.php,
|
|
|
|
* and only be called once in your code.
|
|
|
|
*
|
|
|
|
* @param Array $searchableClasses The extension will be applied to all DataObject subclasses
|
|
|
|
* listed here. Default: {@link SiteTree} and {@link File}.
|
2010-10-04 04:32:48 +00:00
|
|
|
*/
|
|
|
|
static function enable($searchableClasses = array('SiteTree', 'File')) {
|
|
|
|
$defaultColumns = array(
|
|
|
|
'SiteTree' => 'Title,MenuTitle,Content,MetaTitle,MetaDescription,MetaKeywords',
|
|
|
|
'File' => 'Filename,Title,Content'
|
|
|
|
);
|
2010-10-19 03:54:51 +00:00
|
|
|
|
2010-10-04 04:32:48 +00:00
|
|
|
if(!is_array($searchableClasses)) $searchableClasses = array($searchableClasses);
|
|
|
|
foreach($searchableClasses as $class) {
|
2011-03-23 16:32:24 +13:00
|
|
|
if(!class_exists($class)) continue;
|
|
|
|
|
2010-10-04 04:32:48 +00:00
|
|
|
if(isset($defaultColumns[$class])) {
|
2011-10-29 16:48:09 +13:00
|
|
|
if(DB::getConn()->getDatabaseServer() == 'mysql') {
|
|
|
|
Object::add_static_var($class, 'create_table_options', array('MySQLDatabase' => 'ENGINE=MyISAM'), true);
|
|
|
|
}
|
2010-10-13 01:26:51 +00:00
|
|
|
Object::add_extension($class, "FulltextSearchable('{$defaultColumns[$class]}')");
|
2010-10-04 04:32:48 +00:00
|
|
|
} else {
|
2010-10-13 01:26:51 +00:00
|
|
|
throw new Exception("FulltextSearchable::enable() I don't know the default search columns for class '$class'");
|
2010-10-04 04:32:48 +00:00
|
|
|
}
|
|
|
|
}
|
2010-10-19 04:53:36 +00:00
|
|
|
self::$searchable_classes = $searchableClasses;
|
2011-03-18 15:01:09 +13:00
|
|
|
if(class_exists("ContentController")){
|
|
|
|
Object::add_extension("ContentController", "ContentControllerSearchExtension");
|
|
|
|
}
|
2010-10-04 04:32:48 +00:00
|
|
|
}
|
2010-10-19 03:54:51 +00:00
|
|
|
|
2010-10-19 04:53:36 +00:00
|
|
|
/**
|
|
|
|
* @param Array|String $searchFields Comma-separated list (or array) of database column names
|
|
|
|
* that can be searched on. Used for generation of the database index defintions.
|
|
|
|
*/
|
2010-10-04 04:32:48 +00:00
|
|
|
function __construct($searchFields) {
|
|
|
|
if(is_array($searchFields)) $this->searchFields = implode(',', $searchFields);
|
|
|
|
else $this->searchFields = $searchFields;
|
2010-10-19 04:53:36 +00:00
|
|
|
|
2010-10-04 04:32:48 +00:00
|
|
|
parent::__construct();
|
|
|
|
}
|
2010-10-19 03:54:51 +00:00
|
|
|
|
2011-10-31 11:17:37 +13:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param string $class
|
|
|
|
* @param string $extension
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
function extraStatics($class=null, $extension=null) {
|
2010-10-04 04:32:48 +00:00
|
|
|
if($extension && preg_match('/\([\'"](.*)[\'"]\)/', $extension, $matches)) {
|
|
|
|
$searchFields = $matches[1];
|
2010-10-19 03:54:51 +00:00
|
|
|
|
2010-10-04 04:32:48 +00:00
|
|
|
return array(
|
|
|
|
'indexes' => array(
|
|
|
|
"SearchFields" => Array(
|
2010-10-19 03:54:51 +00:00
|
|
|
'type'=>'fulltext',
|
|
|
|
'name'=>'SearchFields',
|
2010-10-04 04:32:48 +00:00
|
|
|
'value'=> $searchFields
|
|
|
|
),
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2010-10-19 04:53:36 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Shows all classes that had the {@link FulltextSearchable} extension applied through {@link enable()}.
|
|
|
|
*
|
|
|
|
* @return Array
|
|
|
|
*/
|
|
|
|
function get_searchable_classes() {
|
|
|
|
return self::$searchable_classes;
|
|
|
|
}
|
|
|
|
|
2011-10-29 17:12:02 +13:00
|
|
|
}
|