Cloning an existing WikiPage

Last edited by JavaWoman:
Modified links pointing to docs server
Mon, 28 Jan 2008 00:15 UTC [diff]


This handler is included in WikkaWiki since version 1.1.6.0
See also:
Documentation: CloneHandlerInfo.


This is the development page for the clone handler.
 



I like to re-use and hate to redesign the wheel... Surely I'm not alone ;-)

Two solutions

Either an action CloneAction or this CloneHandler.
When the action was first published, DarTar explained that a handler would make more sense. It certainly is the right way to do so here is the handler.

Dependancy

This relies on the ExistsPage() function developed by JavaWoman and part of release 1.1.6.0. You can find this version at WikkaDevelopment.

The code

Copy the code into a handler named: clone.php - place it in the handlers/page folder.
Thanks to DarTar, the original code has been improved: please get the correct code below.

How to use it?

Once you have (re)designed a page that fits to be replicated many times (Task Lists, User Pages, Bug Description, Developement Description...) you just have to type in /clone at the end of its URL.
Then you must fill the name of the new WikiPage to be created, you may want to add a note, finally you decide if you just want to create the page or if you want to jump and edit it as soon as it has been copied.

To Do

My code needs probably to be reviewed by expert coder as I am not at all a developper.
Any ideas around this handler more than welcome. Here are a few ideas I have to improve this handler:

All discussions on this page still are at WikiTemplate. We definitely need a PageNameChange handler ;-)
See MovePages - not implemented, it has its problems; but the wish/requirement isn't new ;-) --JavaWoman


Improved CloneHandler


Christian, your code had a number of bugs that I've fixed in a new version of the handler posted below.
I think this is an issue that should be addressed in general, not only in this case. We should try to produce an accessible code also for users whose browser does not support javascript. A central error handler, displaying alert messages (both local, as those produced by handlers and actions, and global, as those produced by the database or the engine itself), would be a major improvement of WikkaWiki.


I've uploaded this handler to the server so please test it and give your feedback here.
-- DarTar

<div class="page">
<?php
/**
 * Clone the current page and save a copy of it as a new page.
 *
 * Usage: append /clone to the URL of the page you want to clone
 *
 * This handler checks the existence of the source page, the validity of the
 * name of the target page to be created, the user's read-access to the source
 * page and write-access to the target page.
 * If the edit option is selected, the user is redirected to the target page for
 * edition immediately after its creation.
 *
 * @package         Handlers
 * @subpackage        
 * @name              clone
 *
 * @author            {@link http://wikka.jsnx.com/ChristianBarthelemy Christian Barthelemy} - original idea and code.
 * @author            {@link http://wikka.jsnx.com/DarTar Dario Taraborelli} - bugs fixed, code improved, removed popup alerts.  
 * @version           0.4
 * @since             Wikka 1.1.6.X
 *                      
 * @input             string  $to  required: the page to be created
 *                            must be a non existing page and current user must be authorized to create it
 *                            default is source page name              
 *
 * @input             string  $note  optional: the note to be added to the page when created
 *                            default is "Cloned from " followed by the name of the source page
 *
 * @input             boolean $editoption optional: if true, the new page will be opened for edition on creation
 *                            default is false (to allow multiple cloning of the same source)
 *
 * @todo              Use central library for valid pagenames.
 *        
 */


// set defaults
$from = $this->tag;
$to = $this->tag;
$note = 'Cloned from '.$from; #i18n
$editoption = '';
$box = 'Please fill in a valid target ""PageName"" and an (optional) edit note.'; #i18n

// print header
echo $this->Format('==== Clone current page ====');

// 1. check source page existence
if (!$this->ExistsPage($from))
{
    // source page does not exist!
    $box = ' Sorry, page '.$from.' does not exist.'; #i18n
} else
{
    // 2. page exists - now check user's read-access to the source page
    if (!$this->HasAccess('read', $from))
    {
        // user can't read source page!
        $box = ' //You aren not allowed to read the source of this page.//'; #i18n
    } else
    {
        // page exists and user has read-access to the source - proceed
        if ($_POST)
        {
            // get parameters
            $to = ($_POST['to'])? $_POST['to'] : $to;
            $note = ($_POST['note'])? $_POST['note'] : $note;
            $editoption = (isset($_POST['editoption']))? 'checked="checked"' : '';
       
            // 3. check target pagename validity
            if (!preg_match("/^[A-ZÄÖÜ]+[a-zßäöü]+[A-Z0-9ÄÖÜ][A-Za-z0-9ÄÖÜßäöü]*$/s", $to))
            {
                // invalid pagename!
                $box = '""<div class="error">You must specify a valid PageName</div>""'; #i18n
            } else
            {
                // 4. target page name is valid - now check user's write-access
                if (!$this->HasAccess('write', $to))  
                {
                    $box = '""<div class="error">Sorry! You don\'t have write-access to '.$to.'</div>""'; #i18n
                } else
                {
                    // 5. check target page existence
                    if ($this->ExistsPage($to))
                    {
                        // page already exists!
                            $box = '""<div class="error">Sorry! The destination page already exists</div>""'; #i18n
                    } else
                    {
                        // 6. Valid request - proceed to page cloning
                        $thepage=$this->LoadPage($from); # load the source page
                        if ($thepage) $pagecontent = $thepage['body']; # get its content
                        $this->SavePage($to, $pagecontent, $note); #create target page
                        if ($editoption == 'checked="checked"')
                        {
                            // quick edit
                            $this->Redirect($this->href('edit',$to));
                        } else
                        {
                            // show confirmation message
                            $box = '""'.$this->MiniHref('',$to).'"" was succesfully created!'; #i18n
                        }
                    }
                }
            }
        }
        // build form
        $form = $this->FormOpen('clone');
        $form .= '<table class="clone">'.
            '<tr>'.
            '<td><strong>Clone '.$this->Link($this->GetPageTag()).' to:</strong></td>'.
            '<td><input type="text" name="to" value="'.$to.'" size="37" /></td>'.
            '</tr>'.
            '<tr>'.
            '<td><strong>Edit note:</strong></td>'.
            '<td><input type="text" name="note" value="'.$note.'" size="37" /></td>'.
            '</tr>'.
            '<tr>'.
            '<td></td>'.
            '<td>'.
            '<input type="checkbox" name="editoption" '.$editoption.' /> Edit after creation '.
            '<input type="submit" name="create" value="Clone" />'.
            '</td>'.
            '</tr>'.
            '</table>';
        $form .= $this->FormClose();
    }
}

// display messages
if (isset($box)) echo $this->Format(' --- '.$box.' --- --- ');
// print form
if (isset($form)) print $form;
?>
</div>


I've added the feature of letting the user clone the ACLs in this version:
(I also fixed what I thought was a bug, it's now possible to have a blank note for the cloned page.)
Comments are welcome.
/AndreasHeintze
<div class="page">
<?php
/**
 * Clone the current page and save a copy of it as a new page.
 *
 * Usage: append /clone to the URL of the page you want to clone
 *
 * This handler checks the existence of the source page, the validity of the
 * name of the target page to be created, the user's read-access to the source
 * page and write-access to the target page.
 * If the edit option is selected, the user is redirected to the target page for
 * edition immediately after its creation.
 *
 * @package         Handlers
 * @subpackage        
 * @name              clone
 *
 * @author            {@link http://wikkawiki.org/ChristianBarthelemy Christian Barthelemy} - original idea and code.
 * @author            {@link http://wikkawiki.org/DarTar Dario Taraborelli} - bugs fixed, code improved, removed popup alerts.  
 * @version           0.4
 * @since             Wikka 1.1.6.0
 *                      
 * @input             string  $to  required: the page to be created
 *                            must be a non existing page and current user must be authorized to create it
 *                            default is source page name              
 *
 * @input             string  $note  optional: the note to be added to the page when created
 *                            default is "Cloned from " followed by the name of the source page
 *
 * @input             boolean $editoption optional: if true, the new page will be opened for edition on creation
 *                            default is false (to allow multiple cloning of the same source)
 *
 * @todo              Use central library for valid pagenames.
 *        
 */

// defaults
if(!defined('VALID_PAGENAME_PATTERN')) define ('VALID_PAGENAME_PATTERN', '/^[A-Za-zÄÖÜßäöü]+[A-Za-z0-9ÄÖÜßäöü]*$/s');

// i18n
define('CLONE_HEADER', '==== Clone current page ====');
define('CLONE_SUCCESSFUL', '%s was succesfully created!');
define('CLONE_X_TO', 'Clone %s to:');
define('CLONED_FROM', 'Cloned from %s');
define('EDIT_NOTE', 'Edit note:');
define('ERROR_ACL_READ', 'You are not allowed to read the source of this page.');
define('ERROR_ACL_WRITE', 'Sorry! You don\'t have write-access to %s');
define('ERROR_INVALID_PAGENAME', 'This page name is invalid. Valid page names must start with a letter and contain only letters and numbers.');
define('ERROR_PAGE_ALREADY_EXIST', 'Sorry, the destination page already exists');
define('ERROR_PAGE_NOT_EXIST', ' Sorry, page %s does not exist.');
define('LABEL_CLONE', 'Clone');
define('LABEL_EDIT_OPTION', ' Edit after creation ');
define('LABEL_CLONEACLS_OPTION', ' Clone ACLs ');
define('PLEASE_FILL_VALID_TARGET', 'Please fill in a valid target ""PageName"" and an (optional) edit note.');

// initialization
$from = $this->tag;
$to = $this->tag;
$note = sprintf(CLONED_FROM, $from);
$editoption = '';
$cloneacls = '';
$box = PLEASE_FILL_VALID_TARGET;

// print header
echo $this->Format(CLONE_HEADER);

// 1. check source page existence
if (!$this->ExistsPage($from))
{
    // source page does not exist!
    $box = sprintf(ERROR_PAGE_NOT_EXIST, $from);
} else
{
    // 2. page exists - now check user's read-access to the source page
    if (!$this->HasAccess('read', $from))
    {
        // user can't read source page!
        $box = ERROR_ACL_READ;
    } else
    {
        // page exists and user has read-access to the source - proceed
        if (isset($_POST) && $_POST)
        {
            // get parameters
            $to = isset($_POST['to']) && $_POST['to'] ? $_POST['to'] : $to;
            $note = isset($_POST['note']) ? $_POST['note'] : $note;
            $editoption = (isset($_POST['editoption']))? 'checked="checked"' : '';
            $cloneacls = (isset($_POST['cloneacls']))? 'checked="checked"' : '';
       
            // 3. check target pagename validity
            if (!preg_match(VALID_PAGENAME_PATTERN, $to))  //TODO use central regex library
            {
                // invalid pagename!
                $box = '""<em class="error">'.ERROR_INVALID_PAGENAME.'</em>""';
            } else
            {
                // 4. target page name is valid - now check user's write-access
                if (!$this->HasAccess('write', $to))  
                {
                    $box = '""<em class="error">'.sprintf(ERROR_ACL_WRITE, $to).'</em>""';
                } else
                {
                    // 5. check target page existence
                    if ($this->ExistsPage($to))
                    {
                        // page already exists!
                        $box = '""<em class="error">'.ERROR_PAGE_ALREADY_EXIST.'</em>""';
                    } else
                    {
                        // 6. Valid request - proceed to page cloning
                        $thepage=$this->LoadPage($from); # load the source page
                        if ($thepage) $pagecontent = $thepage['body']; # get its content
                        $this->SavePage($to, $pagecontent, $note); #create target page
                        if ($cloneacls == 'checked="checked"')
                        {
                            // Clone ACLs too
                            $acls = $this->LoadAllACLs($from);
                            $this->SaveACL($to, 'read', $acls['read_acl']);
                            $this->SaveACL($to, 'write', $acls['write_acl']);
                            $this->SaveACL($to, 'comment', $acls['comment_acl']);
                        }
                        if ($editoption == 'checked="checked"')
                        {
                            // quick edit
                            $this->Redirect($this->href('edit', $to));
                        } else
                        {
                            // show confirmation message
                            $box = '""<em class="success">'.sprintf(CLONE_SUCCESSFUL, $to).'</em>""';
                        }
                    }
                }
            }
        }
        // build form
        $form = $this->FormOpen('clone');
        $form .= '<table class="clone">'."\n".
            '<tr>'."\n".
            '<td>'.sprintf(CLONE_X_TO, $this->Link($this->GetPageTag())).'</td>'."\n".
            '<td><input type="text" name="to" value="'.$to.'" size="37" maxlength="75" /></td>'."\n".
            '</tr>'."\n".
            '<tr>'."\n".
            '<td>'.EDIT_NOTE.'</strong></td>'.
            '<td><input type="text" name="note" value="'.$note.'" size="37" maxlength="75" /></td>'."\n".
            '</tr>'."\n".
            '<tr>'."\n".
            '<td></td>'."\n".
            '<td>'."\n".
            '<input type="checkbox" name="editoption" '.$editoption.' id="editoption" /><label for="editoption">'.LABEL_EDIT_OPTION.'</label><br />'."\n".
            '<input type="checkbox" name="cloneacls" '.$cloneacls.' id="cloneacls" checked /><label for="cloneacls">'.LABEL_CLONEACLS_OPTION.'</label><br /><br />'."\n".
            '<input type="submit" name="create" value="'.LABEL_CLONE.'" />'."\n".
            '</td>'."\n".
            '</tr>'."\n".
            '</table>'."\n";
        $form .= $this->FormClose();
    }
}

// display messages
if (isset($box)) echo $this->Format(' --- '.$box.' --- --- ');
// print form
if (isset($form)) print $form;
?>
</div>




CategoryDevelopmentHandlers
There are 14 comments on this page. [Show comments]
Valid XHTML :: Valid CSS: :: Powered by WikkaWiki