formatted; * \endcode * * An extension is something which is capable of reacting to events. They * register themselves using the add_event_listener() function, after which * events will be sent to the object's recieve_event() function. * * SimpleExtension subclasses are slightly different -- they are registered * automatically, and events are sent to a named method, eg PageRequestEvent * will be sent to onPageRequest() * * * \page hello The Hello World Extension * * \code * // ext/hello/main.php * public class Hello extends SimpleExtension { * public void onPageRequest(PageRequestEvent $event) { * global $page, $user; * $this->theme->display_hello($page, $user); * } * } * * // ext/hello/theme.php * public class HelloTheme extends Themelet { * public void display_hello(Page $page, User $user) { * $page->add_block(new Block("Hello!", "Hello there ".html_escape($user->name)); * } * } * * // ext/hello/test.php * public class HelloTest extends SCoreWebTestCase { * public void testHello() { * $this->get_page("post/list"); * $this->assert_text("Hello there"); * } * } * * // themes/mytheme/hello.theme.php * public class CustomHelloTheme extends HelloTheme { * public function display_hello(Page $page, User $user) { * $h_user = html_escape($user->name); * $page->add_block(new Block( * "Hello!", * "Hello there $h_user, look at my snazzy custom theme!" * ); * } * } * \endcode * */ /** * A generic extension class, for subclassing */ interface Extension { public function receive_event(Event $event); } /** * send_event(BlahEvent()) -> onBlah($event) * * Also loads the theme object into $this->theme if available * * index.php will load all SimpleExtension subclasses, * so no need for register_extension(new Foo()) * * Automatic registration is done with priority returned by get_priority() * * Hopefully this removes as much copy & paste code from the extension * files as possible~ * * The original concept came from Artanis's SimpleExtension extension * --> http://github.com/Artanis/simple-extension/tree/master * Then re-implemented by Shish after he broke the forum and couldn't * find the thread where the original was posted >_< */ abstract class SimpleExtension implements Extension { var $theme; var $_child; public function i_am($child) { $this->_child = $child; if(is_null($this->theme)) $this->theme = get_theme_object($child, false); } public function receive_event(Event $event) { $name = get_class($event); $name = "on".str_replace("Event", "", $name); if(method_exists($this->_child, $name)) { $this->_child->$name($event); } } public function get_priority() {return 50;} } /** * Several extensions have this in common, make a common API */ abstract class FormatterExtension implements Extension { public function receive_event(Event $event) { if($event instanceof TextFormattingEvent) { $event->formatted = $this->format($event->formatted); $event->stripped = $this->strip($event->stripped); } } abstract public function format($text); abstract public function strip($text); } /** * This too is a common class of extension with many methods in common, * so we have a base class to extend from */ abstract class DataHandlerExtension implements Extension { var $theme; public function receive_event(Event $event) { if(is_null($this->theme)) $this->theme = get_theme_object($this); if($event instanceof DataUploadEvent) { $valid=FALSE; // TRUE = skip check_contents() $USEHEADERSALWAYS=FALSE; //TRUE; // STUB - make option if ( is_null($event->type) || ($event->type == "") || ( $USEHEADERSALWAYS ) ) { if (is_callable(array($this,'find_ext'))) { // find_ext should always return false or a valid str, so: if(!$ext=$this->find_ext($event->tmpname)) return; // pass off to the next listener and avoid ext=NULL mysql insert $valid=TRUE; // or avoid calling check_contents() needlessly $event->type = $ext; $event->metadata['extension']=$ext; log_info("core-extension", get_class($this)." has renamed '".$event->metadata['filename']."' to '".$event->metadata['filename'].".$ext'"); $event->metadata['filename']=$event->metadata['filename'].".$ext"; } else { if(!$USEHEADERSALWAYS) return; } // no ext + valid check_contents = mysql error. } if( ($valid) || ($this->check_contents($event->tmpname)) ) { if(!move_upload_to_archive($event)) return; send_event(new ThumbnailGenerationEvent($event->hash, $event->type)); $image = $this->create_image_from_data(warehouse_path("images", $event->hash), $event->metadata); if(is_null($image)) { throw new UploadException("Data handler failed to create image object from data"); } $iae = new ImageAdditionEvent($event->user, $image); send_event($iae); $event->image_id = $iae->image->id; } } if(($event instanceof ThumbnailGenerationEvent) && $this->supported_ext($event->type)) { $this->create_thumb($event->hash); } if(($event instanceof DisplayingImageEvent) && $this->supported_ext($event->image->ext)) { global $page; $this->theme->display_image($page, $event->image); } } abstract protected function supported_ext($ext); abstract protected function check_contents($tmpname); abstract protected function create_image_from_data($filename, $metadata); abstract protected function create_thumb($hash); } ?>