mirror of
				https://github.com/gtbu/Typesetter-5.3-p8.git
				synced 2025-11-04 06:48:27 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			1738 lines
		
	
	
	
		
			47 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			1738 lines
		
	
	
	
		
			47 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						|
 | 
						|
namespace gp\Page;
 | 
						|
 | 
						|
defined('is_running') or die('Not an entry point...');
 | 
						|
 | 
						|
class Edit extends \gp\Page{
 | 
						|
 | 
						|
	protected	$draft_file;
 | 
						|
	protected	$draft_exists = false;
 | 
						|
 | 
						|
	protected	$permission_edit;
 | 
						|
	protected	$permission_menu;
 | 
						|
 | 
						|
	private		$checksum;
 | 
						|
 | 
						|
 | 
						|
	public function __construct($title, $type){
 | 
						|
		parent::__construct($title, $type);
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
	public function RunScript(){
 | 
						|
		global $langmessage;
 | 
						|
 | 
						|
		if( !$this->SetVars() ){
 | 
						|
			return;
 | 
						|
		}
 | 
						|
 | 
						|
		ob_start();
 | 
						|
 | 
						|
		$this->GetFile();
 | 
						|
		$cmd = \gp\tool::GetCommand();
 | 
						|
		$this->RunCommands($cmd);
 | 
						|
 | 
						|
		$this->contentBuffer .= ob_get_clean();
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Run Commands
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public function RunCommands($cmd){
 | 
						|
 | 
						|
		//allow addons to effect page actions and how a page is displayed
 | 
						|
		$cmd = \gp\tool\Plugins::Filter('PageRunScript', [$cmd]);
 | 
						|
		if( $cmd === 'return' ){
 | 
						|
			return;
 | 
						|
		}
 | 
						|
 | 
						|
		parent::RunCommands($cmd);
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * SetVars
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public function SetVars(){
 | 
						|
		global $dataDir;
 | 
						|
 | 
						|
		if( !parent::SetVars() ){
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
 | 
						|
		$this->permission_edit	= \gp\admin\Tools::CanEdit($this->gp_index);
 | 
						|
		$this->permission_menu	= \gp\admin\Tools::HasPermission('Admin_Menu');
 | 
						|
		$this->draft_file		= dirname($this->file) . '/draft.php';
 | 
						|
 | 
						|
		//admin actions
 | 
						|
		if( $this->permission_menu ){
 | 
						|
			$this->cmds['RenameForm']				= '\\gp\\Page\\Rename::RenameForm';
 | 
						|
			$this->cmds['RenameFile']				= '\\gp\\Page\\Rename::RenamePage';
 | 
						|
			$this->cmds['ToggleVisibility']			= ['\\gp\\Page\\Visibility::TogglePage', 'DefaultDisplay'];
 | 
						|
		}
 | 
						|
 | 
						|
		if( $this->permission_edit ){
 | 
						|
			/* gallery/image editing */
 | 
						|
			$this->cmds['Gallery_Folder']			= 'GalleryImages';
 | 
						|
			$this->cmds['Gallery_Images']			= 'GalleryImages';
 | 
						|
			$this->cmds['Image_Editor']				= '\\gp\\tool\\Editing::ImageEditor';
 | 
						|
			$this->cmds['New_Dir']					= '\\gp\\tool\\Editing::NewDirForm';
 | 
						|
 | 
						|
			$this->cmds['ManageSections']			= '';
 | 
						|
			$this->cmds['SaveSections']				= '';
 | 
						|
			$this->cmds['SaveToClipboard']			= '';
 | 
						|
			$this->cmds['RemoveFromClipboard']		= '';
 | 
						|
			$this->cmds['RelabelClipboardItem']		= '';
 | 
						|
			$this->cmds['ReorderClipboardItems']	= '';
 | 
						|
			$this->cmds['AddFromClipboard']			= '';
 | 
						|
 | 
						|
			$this->cmds['ViewRevision']				= '';
 | 
						|
			$this->cmds['ViewCurrent']				= '';
 | 
						|
			$this->cmds['PublishDraft']				= 'DefaultDisplay';
 | 
						|
 | 
						|
			/* inline editing */
 | 
						|
			$this->cmds['Save']						= 'SectionEdit';
 | 
						|
			$this->cmds['Save_Inline']				= 'SectionEdit';
 | 
						|
			$this->cmds['Preview']					= 'SectionEdit';
 | 
						|
			$this->cmds['Include_Dialog']			= 'SectionEdit';
 | 
						|
			$this->cmds['InlineEdit']				= 'SectionEdit';
 | 
						|
		}
 | 
						|
 | 
						|
		if( !\gp\tool\Files::Exists($this->draft_file) ){
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
 | 
						|
		$this->draft_exists = true;
 | 
						|
 | 
						|
		return true;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Display after commands have been executed
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public function DefaultDisplay(){
 | 
						|
 | 
						|
		//add to all pages in case a user adds a gallery
 | 
						|
		\gp\tool\Plugins::Action('GenerateContent_Admin');
 | 
						|
		\gp\tool::ShowingGallery();
 | 
						|
 | 
						|
		$sections_count			= count($this->file_sections);
 | 
						|
		$this->file_sections	= array_values($this->file_sections);
 | 
						|
		$section_num			= 0;
 | 
						|
 | 
						|
		while( $section_num < $sections_count ){
 | 
						|
			echo $this->GetSection($section_num);
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Get the data file, get draft file if it exists
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public function GetFile(){
 | 
						|
 | 
						|
		parent::GetFile();
 | 
						|
 | 
						|
		if( $this->draft_exists ){
 | 
						|
			$this->file_sections	= \gp\tool\Files::Get($this->draft_file, 'file_sections');
 | 
						|
			$this->meta_data		= \gp\tool\Files::$last_meta;
 | 
						|
			$this->file_stats		= \gp\tool\Files::$last_stats;
 | 
						|
		}
 | 
						|
 | 
						|
		$this->checksum				= $this->Checksum();
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Generate admin toolbar links
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public function AdminLinks(){
 | 
						|
		global $langmessage, $config;
 | 
						|
 | 
						|
		$admin_links = [];
 | 
						|
 | 
						|
		// HideAdminUI
 | 
						|
		$admin_links[] = \gp\tool::Link(
 | 
						|
			$this->title,
 | 
						|
			'<i class="fa fa-eye-slash"></i>',
 | 
						|
			'',
 | 
						|
			[
 | 
						|
				'title'		=> $langmessage['Hide Admin UI'],
 | 
						|
				'class'		=> 'admin-link admin-link-hide-ui',
 | 
						|
				'data-cmd'	=> 'hide_ui',
 | 
						|
			]
 | 
						|
		);
 | 
						|
 | 
						|
		// page options: less frequently used links that don't have to do with editing the content of the page
 | 
						|
		$option_links = [];
 | 
						|
		if( $this->permission_menu ){
 | 
						|
			$option_links[] = \gp\tool::Link(
 | 
						|
				$this->title,
 | 
						|
				$langmessage['rename/details'],
 | 
						|
				'cmd=renameform&index=' . urlencode($this->gp_index),
 | 
						|
				[
 | 
						|
					'class'		=> 'admin-link admin-link-rename-details',
 | 
						|
					'data-cmd'	=> 'gpajax',
 | 
						|
				]
 | 
						|
			);
 | 
						|
			$option_links[] = \gp\tool::Link(
 | 
						|
				'Admin/Menu',
 | 
						|
				$langmessage['current_layout'],
 | 
						|
				'cmd=layout&from=page&index=' . urlencode($this->gp_index),
 | 
						|
				[
 | 
						|
					'title'		=> $langmessage['current_layout'],
 | 
						|
					'class'		=> 'admin-link admin-link-current-laout',
 | 
						|
					'data-cmd'	=> 'gpabox',
 | 
						|
				]
 | 
						|
			);
 | 
						|
			$option_links[] = \gp\tool::Link(
 | 
						|
				'Admin/Menu/Ajax',
 | 
						|
				$langmessage['Copy'],
 | 
						|
				'cmd=CopyForm&redir=redir&index=' . urlencode($this->gp_index),
 | 
						|
				[
 | 
						|
					'title'		=> $langmessage['Copy'],
 | 
						|
					'class'		=> 'admin-link admin-link-copy-page',
 | 
						|
					'data-cmd'	=> 'gpabox'
 | 
						|
				]
 | 
						|
			);
 | 
						|
		}
 | 
						|
 | 
						|
		if( \gp\admin\Tools::HasPermission('Admin_User') ){
 | 
						|
			$option_links[] = \gp\tool::Link(
 | 
						|
				'Admin/Permissions',
 | 
						|
				$langmessage['permissions'],
 | 
						|
				'index=' . urlencode($this->gp_index),
 | 
						|
				[
 | 
						|
					'title'		=> $langmessage['permissions'],
 | 
						|
					'class'		=> 'admin-link admin-link-permissions',
 | 
						|
					'data-cmd'	=> 'gpabox',
 | 
						|
				]
 | 
						|
			);
 | 
						|
		}
 | 
						|
 | 
						|
		if( $this->permission_menu ){
 | 
						|
			$option_links[]	= self::ToggleVisibilityLink($this->gp_index, $this->visibility != 'private');
 | 
						|
		}
 | 
						|
 | 
						|
		if( $this->permission_menu ){
 | 
						|
			$option_links[] = \gp\tool::Link(
 | 
						|
				'Admin/Menu/Ajax',
 | 
						|
				$langmessage['delete_file'],
 | 
						|
				'cmd=MoveToTrash&index=' . urlencode($this->gp_index),
 | 
						|
				[
 | 
						|
					'title'		=> $langmessage['delete_page'],
 | 
						|
					'class'		=>'gpconfirm admin-link admin-link-delete-page',
 | 
						|
					'data-cmd'	=> 'postlink',
 | 
						|
				]
 | 
						|
			);
 | 
						|
		}
 | 
						|
 | 
						|
		if( !empty($option_links) ){
 | 
						|
			$admin_links[$langmessage['options']] = $option_links;
 | 
						|
		}
 | 
						|
 | 
						|
		//publish draft
 | 
						|
		if( $this->permission_edit && $this->draft_exists ){
 | 
						|
			$admin_links[] = \gp\tool::Link(
 | 
						|
				$this->title,
 | 
						|
				'<i class="fa fa-check"></i> ' . $langmessage['Publish Draft'],
 | 
						|
				'cmd=PublishDraft',
 | 
						|
				[
 | 
						|
					'class'		=> 'msg_publish_draft admin-link admin-link-publish-draft',
 | 
						|
					'data-cmd'	=> 'creq',
 | 
						|
				]
 | 
						|
			)
 | 
						|
			. '<a class="msg_publish_draft_disabled admin-link admin-link-publish-draft-disabled">'
 | 
						|
			.   '<i class="fa fa-minus-circle"></i> ' . $langmessage['Publish Draft']
 | 
						|
			. '</a>'
 | 
						|
			. '<a class="msg_saving_draft admin-link admin-link-publish-draft-saving">'
 | 
						|
			.   '<i class="fa fa-spinner fa-pulse"></i> ' . $langmessage['Saving'] . ' …'
 | 
						|
			. '</a>';
 | 
						|
		}
 | 
						|
 | 
						|
		return array_merge($admin_links, $this->admin_links);
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Return Toggle Page Visibility option link
 | 
						|
	 * @param string page index
 | 
						|
	 * @param bool $visisbility of current page
 | 
						|
	 * @return string formatted link
 | 
						|
	 */
 | 
						|
	public static function ToggleVisibilityLink($index, $visibility){
 | 
						|
		global $langmessage;
 | 
						|
		$q				= 'cmd=ToggleVisibility&index=' . urlencode($index);
 | 
						|
		$label			= $langmessage['Visibility'] . ': ' . $langmessage['Private'];
 | 
						|
		$attrs			= [
 | 
						|
							'class'		=> 'admin-link admin-link-toggle-visibility',
 | 
						|
							'data-cmd'	=> 'postlink',
 | 
						|
						];
 | 
						|
		if( $visibility ){
 | 
						|
			$q			.= '&visibility=private';
 | 
						|
			$label		= $langmessage['Visibility'] . ': ' . $langmessage['Public'];
 | 
						|
		}else{
 | 
						|
			$attrs['class'] .= ' admin-link-visibility-private';
 | 
						|
		}
 | 
						|
		return \gp\tool::Link('Admin/Menu/Ajax', $label, $q, $attrs);
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Send js to client for managing content sections
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public static function ManageSections(){
 | 
						|
		global $langmessage, $page;
 | 
						|
 | 
						|
		$scripts = [];
 | 
						|
 | 
						|
		//output links
 | 
						|
		ob_start();
 | 
						|
		if( $page->pagetype == 'display' ){
 | 
						|
			echo '<div id="section_sorting_wrap" class="inline_edit_area">';
 | 
						|
			echo '<ul id="section_sorting" class="section_drag_area" title="Organize"></ul>';
 | 
						|
 | 
						|
			echo '<div id="section-clipboard">';
 | 
						|
			echo '<ul id="section-clipboard-items">';
 | 
						|
			echo self::SectionClipboardLinks();
 | 
						|
			echo '</ul>';
 | 
						|
			echo '</div>';
 | 
						|
 | 
						|
			// echo '<div>'.$langmessage['add'].'</div>';
 | 
						|
			echo '<div id="new_section_links">';
 | 
						|
			self::NewSections();
 | 
						|
			echo '</div>';
 | 
						|
			echo '</div>';
 | 
						|
		}
 | 
						|
 | 
						|
		echo '<div id="ck_editable_areas" class="inline_edit_area">';
 | 
						|
		echo '<ul></ul>';
 | 
						|
		echo '</div>';
 | 
						|
 | 
						|
		$scripts[]				= ['code' => 'var section_types = ' . json_encode(ob_get_clean()) . ';'];
 | 
						|
 | 
						|
		// selectable classes moved to \gp\tool\Output::GetHead_InlineJS() because we have more info there
 | 
						|
		// $avail_classes			= \gp\admin\Settings\Classes::GetClasses();
 | 
						|
		// $avail_classes			= \gp\tool\Plugins::Filter('AvailableClasses', [$avail_classes]);
 | 
						|
		// $scripts[]				= ['code' => 'var gp_avail_classes = ' . json_encode($avail_classes) . ';'];
 | 
						|
 | 
						|
		$scripts[]				= ['object' => 'gp_editing', 'file' => '/include/js/inline_edit/inline_editing.js'];
 | 
						|
 | 
						|
		if( empty($_REQUEST['mode']) ){
 | 
						|
			$scripts[]			= ['object' => 'gp_editing', 'code' => 'gp_editing.is_extra_mode = false;'];
 | 
						|
		}else{
 | 
						|
			$scripts[]			= ['object' => 'gp_editing', 'code' => 'gp_editing.is_extra_mode = true;'];
 | 
						|
		}
 | 
						|
 | 
						|
		$scripts[]				= ['file' => '/include/js/inline_edit/manage_sections.js'];
 | 
						|
 | 
						|
		\gp\tool\Output\Ajax::SendScripts($scripts);
 | 
						|
		die();
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Send multiple sections to the client
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public function NewNestedSection($types, $wrapper_data){
 | 
						|
		global $langmessage;
 | 
						|
 | 
						|
		$new_section = \gp\tool\Editing::DefaultContent('wrapper_section');
 | 
						|
 | 
						|
		if( is_array($wrapper_data) ){
 | 
						|
			// Typesetter > 5.0.3: $wrapper_data may be defined as array by plugins
 | 
						|
			$new_section = array_merge($new_section, $wrapper_data);
 | 
						|
		}else{
 | 
						|
			// Typesetter <= 5.0.3: $wrapper_data is a string (wrapper class)
 | 
						|
			$new_section['attributes']['class'] .= ' ' . $wrapper_data;
 | 
						|
		}
 | 
						|
		$orig_attrs = $new_section['attributes'];
 | 
						|
 | 
						|
		$output = $this->SectionNode($new_section, $orig_attrs);
 | 
						|
		foreach($types as $type){
 | 
						|
			if( is_array($type) ){
 | 
						|
				$_wrapper_data = isset($type[1]) ? $type[1] : '';
 | 
						|
				$output .= $this->NewNestedSection($type[0], $_wrapper_data);
 | 
						|
			}else{
 | 
						|
				$output .= $this->GetNewSection($type);
 | 
						|
			}
 | 
						|
 | 
						|
		}
 | 
						|
		$output .= '</div>';
 | 
						|
 | 
						|
		return $output;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	public function GetNewSection($type){
 | 
						|
 | 
						|
		$class			= self::TypeClass($type);
 | 
						|
		$num			= time().rand(0, 10000);
 | 
						|
		$new_section	= \gp\tool\Editing::DefaultContent($type);
 | 
						|
		$content		= \gp\tool\Output\Sections::RenderSection($new_section, $num, $this->title, $this->file_stats);
 | 
						|
 | 
						|
		$new_section['attributes']['class']	.= ' ' . $class;
 | 
						|
		$new_section['gp_type']				= $type;
 | 
						|
		$orig_attrs							= $new_section['attributes'];
 | 
						|
 | 
						|
		if( !isset($new_section['nodeName']) ){
 | 
						|
			return $this->SectionNode($new_section, $orig_attrs) . $content . '</div>';
 | 
						|
		}
 | 
						|
 | 
						|
		return $this->SectionNode($new_section, $orig_attrs) .
 | 
						|
			$content .
 | 
						|
			\gp\tool\Output\Sections::EndTag($new_section['nodeName']);
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	public function SectionNode($section, $orig_attrs){
 | 
						|
 | 
						|
		//if image type, make sure the src is a complete path
 | 
						|
		if( $section['type'] == 'image' ){
 | 
						|
			$orig_attrs['src'] = \gp\tool::GetDir($orig_attrs['src']);
 | 
						|
		}
 | 
						|
 | 
						|
		$orig_attrs			= json_encode($orig_attrs);
 | 
						|
 | 
						|
		if( \gp\tool\Output::ShowEditLink() && \gp\admin\Tools::CanEdit($this->gp_index) ){
 | 
						|
			$section['attributes']['class']		.= ' editable_area';
 | 
						|
		}
 | 
						|
 | 
						|
		$section_attrs		= ['gp_label', 'gp_color', 'gp_collapse', 'gp_type', 'gp_hidden', 'gp_file_under'];
 | 
						|
		foreach($section_attrs as $attr){
 | 
						|
			if( !empty($section[$attr]) ){
 | 
						|
				$section['attributes']['data-' . $attr] = $section[$attr];
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		$attributes			= \gp\tool\Output\Sections::SectionAttributes($section['attributes'], $section['type']);
 | 
						|
		$attributes			.= ' data-gp-attrs=\'' . htmlspecialchars($orig_attrs, ENT_QUOTES & ~ENT_COMPAT) . '\'';
 | 
						|
 | 
						|
 | 
						|
		if( !isset($section['nodeName']) ){
 | 
						|
			return '<div' . $attributes . '>';
 | 
						|
		}
 | 
						|
 | 
						|
		return '<' . $section['nodeName'] . $attributes . '>';
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Save new/rearranged sections
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public function SaveSections(){
 | 
						|
		global $langmessage, $dataDir;
 | 
						|
 | 
						|
		if( isset($_POST['sections_json']) ){
 | 
						|
			// section data is posted as JSON
 | 
						|
			$section_data = json_decode(rawurldecode($_POST['sections_json']), true);
 | 
						|
			$_POST['section_data_is'] = 'json';
 | 
						|
			$_POST += $section_data;
 | 
						|
			unset($_POST['sections_json']);
 | 
						|
		}
 | 
						|
 | 
						|
		$this->ajaxReplace		= [];
 | 
						|
		$original_sections		= $this->file_sections;
 | 
						|
		$unused_sections		= $this->file_sections; //keep track of sections that aren't used
 | 
						|
		$new_sections			= [];
 | 
						|
 | 
						|
		//make sure section_order isn't empty
 | 
						|
		if( empty($_POST['section_order']) ){
 | 
						|
			msg($langmessage['OOPS'] . ' (Invalid Request)');
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
 | 
						|
		foreach($_POST['section_order'] as $i => $arg ){
 | 
						|
			$new_section 		= $this->SaveSection($i, $arg, $unused_sections);
 | 
						|
			if( $new_section === false ){
 | 
						|
				return false;
 | 
						|
			}
 | 
						|
			$new_sections[$i] = $new_section;
 | 
						|
		}
 | 
						|
 | 
						|
		//make sure there's at least one section
 | 
						|
		if( empty($new_sections) ){
 | 
						|
			msg($langmessage['OOPS'] . ' (1 Section Minimum)');
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
 | 
						|
		$this->file_sections = array_values($new_sections);
 | 
						|
 | 
						|
		// save a send message to user
 | 
						|
		if( !$this->SaveThis() ){
 | 
						|
			$this->file_sections = $original_sections;
 | 
						|
			msg($langmessage['OOPS'] . '(4)');
 | 
						|
			return;
 | 
						|
		}
 | 
						|
 | 
						|
		$this->ajaxReplace[] = ['ck_saved', '', ''];
 | 
						|
 | 
						|
		//update gallery info
 | 
						|
		$this->GalleryEdited();
 | 
						|
 | 
						|
		//update usage of resized images
 | 
						|
		foreach($unused_sections as $section_data){
 | 
						|
			if( isset($section_data['resized_imgs']) ){
 | 
						|
				includeFile('image.php');
 | 
						|
				\gp_resized::SetIndex();
 | 
						|
				\gp\tool\Editing::ResizedImageUse($section_data['resized_imgs'], []);
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	protected function SaveSection($i, $arg, &$unused_sections){
 | 
						|
		global $langmessage;
 | 
						|
 | 
						|
		$section_attrs			= ['gp_label', 'gp_color', 'gp_collapse', 'gp_type', 'gp_hidden', 'gp_file_under'];
 | 
						|
 | 
						|
		// moved / copied sections
 | 
						|
		if( ctype_digit($arg) ){
 | 
						|
			$arg = (int)$arg;
 | 
						|
 | 
						|
			if( !isset($this->file_sections[$arg]) ){
 | 
						|
				msg($langmessage['OOPS'] . ' (Invalid Section Number)');
 | 
						|
				return false;
 | 
						|
			}
 | 
						|
 | 
						|
			unset($unused_sections[$arg]);
 | 
						|
			$new_section				= $this->file_sections[$arg];
 | 
						|
			$new_section['attributes']	= [];
 | 
						|
 | 
						|
		// otherwise, new sections
 | 
						|
		}else{
 | 
						|
			$new_section	= \gp\tool\Editing::DefaultContent($arg);
 | 
						|
		}
 | 
						|
 | 
						|
		// attributes
 | 
						|
		$this->PostedAttributes($new_section,$i);
 | 
						|
 | 
						|
		// wrapper section 'contains_sections'
 | 
						|
		if( $new_section['type'] == 'wrapper_section' ){
 | 
						|
			$new_section['contains_sections'] = isset($_POST['contains_sections']) ? $_POST['contains_sections'][$i] : '0';
 | 
						|
		}
 | 
						|
 | 
						|
		// section attributes
 | 
						|
		foreach($section_attrs as $attr){
 | 
						|
			unset($new_section[$attr]);
 | 
						|
			if( !empty($_POST[$attr][$i]) ){
 | 
						|
				$new_section[$attr]		= $_POST[$attr][$i];
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		return $new_section;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Save a section or wrapper with sections to the Clipboard
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	protected function SaveToClipboard(){
 | 
						|
		global $langmessage;
 | 
						|
		$this->ajaxReplace = [];
 | 
						|
 | 
						|
		if( !isset($_POST['section_number']) || !ctype_digit($_POST['section_number']) ){
 | 
						|
			msg($langmessage['OOPS'] . ' (SaveToClipboard: Invalid Request)');
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
 | 
						|
		$section_num = $_POST['section_number'];
 | 
						|
		if( !isset($this->file_sections[$section_num]) ){
 | 
						|
			msg($langmessage['OOPS'] . ' (SaveToClipboard: Invalid Section Number (' . $section_num . ')');
 | 
						|
		}
 | 
						|
 | 
						|
		$file_sections = self::ExtractSections($section_num);
 | 
						|
		// msg('SaveToClipboard: $file_sections = ' . pre($file_sections) );
 | 
						|
 | 
						|
		$first_section = $file_sections[0];
 | 
						|
		$new_clipboard_item = [
 | 
						|
			'type'			=> $first_section['type'],
 | 
						|
			'label'			=> isset($first_section['gp_label'])		? $first_section['gp_label']		:	ucfirst($first_section['type']),
 | 
						|
			'color'			=> isset($first_section['gp_color'])		? $first_section['gp_color']		:	'#aabbcc',
 | 
						|
			'hidden'		=> isset($first_section['gp_hidden'])		? $first_section['gp_hidden']		:	false,
 | 
						|
			'hidden'		=> isset($first_section['gp_file_under'])	? $first_section['gp_file_under']	:	'',
 | 
						|
			'content'		=> $this->GetSectionForClipboard($section_num),
 | 
						|
			'file_sections'	=> $file_sections,
 | 
						|
		];
 | 
						|
 | 
						|
		$clipboard_data = \gp\tool\Files::GetSectionClipboard();
 | 
						|
		// msg("GetSectionClipboard returns " . pre($clipboard_data));
 | 
						|
		array_unshift($clipboard_data, $new_clipboard_item);
 | 
						|
		if( \gp\tool\Files::SaveSectionClipboard($clipboard_data) ){
 | 
						|
			// msg($langmessage['SAVED']);
 | 
						|
		}else{
 | 
						|
			msg($langmessage['OOPS'] . ' (Section Clipboard: Save data failed)');
 | 
						|
		}
 | 
						|
 | 
						|
		$clipboard_links = self::SectionClipboardLinks($clipboard_data);
 | 
						|
		$this->ajaxReplace[] = ['inner', '#section-clipboard-items', $clipboard_links];
 | 
						|
		$this->ajaxReplace[] = ['clipboard_init', '', ''];
 | 
						|
		$this->ajaxReplace[] = ['loaded', '', ''];
 | 
						|
 | 
						|
		return true;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Add (append) a Clipboard Item to the page's file sections
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	protected function AddFromClipboard($item_num=false){
 | 
						|
		global $langmessage;
 | 
						|
		$this->ajaxReplace = [];
 | 
						|
 | 
						|
		if( !$item_num ){
 | 
						|
			if( !isset($_POST['item_number']) || !ctype_digit($_POST['item_number']) ){
 | 
						|
				msg($langmessage['OOPS'] . ' (Section Clipboard - Add Item: Invalid Request');
 | 
						|
				return false;
 | 
						|
			}
 | 
						|
			$item_num = $_POST['item_number'];
 | 
						|
		}
 | 
						|
 | 
						|
		$clipboard_data = \gp\tool\Files::GetSectionClipboard();
 | 
						|
 | 
						|
		if( !isset($clipboard_data[$item_num]) ){
 | 
						|
			msg($langmessage['OOPS'] . ' (Section Clipboard - Add Item: Invalid Item Number (' . $item_num . ')');
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
 | 
						|
		$clipboard_sections		= $clipboard_data[$item_num]['file_sections'];
 | 
						|
		$this->file_sections	= array_merge($this->file_sections, $clipboard_sections);
 | 
						|
 | 
						|
		if( !$this->SaveThis() ){
 | 
						|
			msg($langmessage['OOPS'] . ' (Section Clipboard - Add Item: Save Page Failed)');
 | 
						|
			$this->ajaxReplace[] = ['loaded', '', ''];
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
 | 
						|
		// $this->ajaxReplace[] = ['ck_saved', '', ''];
 | 
						|
 | 
						|
		// include updated Admin Toolbar
 | 
						|
		ob_start();
 | 
						|
		\gp\admin\Tools::AdminToolbar();
 | 
						|
		$admin_toolbar = ob_get_clean();
 | 
						|
		if( !empty($admin_toolbar) ){
 | 
						|
			$this->ajaxReplace[] = ['replace', '#admincontent_panel', $admin_toolbar];
 | 
						|
		}
 | 
						|
		$this->ajaxReplace[] = ['loaded', '', ''];
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Remove a Clipboard Item
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	protected function RemoveFromClipboard($item_num=false){
 | 
						|
		global $langmessage;
 | 
						|
		$this->ajaxReplace = [];
 | 
						|
 | 
						|
		if( !$item_num ){
 | 
						|
			if( !isset($_POST['item_number']) || !ctype_digit($_POST['item_number']) ){
 | 
						|
				msg($langmessage['OOPS'] . ' (Section Clipboard - Remove Item: Invalid Request');
 | 
						|
				return false;
 | 
						|
			}
 | 
						|
			$item_num = $_POST['item_number'];
 | 
						|
		}
 | 
						|
 | 
						|
		$clipboard_data = \gp\tool\Files::GetSectionClipboard();
 | 
						|
 | 
						|
		if( !isset($clipboard_data[$item_num]) ){
 | 
						|
			msg($langmessage['OOPS'] . ' (Section Clipboard - Remove Item: Invalid Item Number (' . $item_num . ')');
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
		unset($clipboard_data[$item_num]);
 | 
						|
		$clipboard_data = array_values($clipboard_data);
 | 
						|
 | 
						|
		if( \gp\tool\Files::SaveSectionClipboard($clipboard_data) ){
 | 
						|
			// msg($langmessage['SAVED']);
 | 
						|
		}else{
 | 
						|
			msg($langmessage['OOPS'] . ' (Section Clipboard - Remove Item: Save data failed)');
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
 | 
						|
		$clipboard_links = self::SectionClipboardLinks($clipboard_data);
 | 
						|
		$this->ajaxReplace[] = ['inner', '#section-clipboard-items', $clipboard_links];
 | 
						|
		$this->ajaxReplace[] = ['clipboard_init', '', ''];
 | 
						|
		$this->ajaxReplace[] = ['loaded', '', ''];
 | 
						|
 | 
						|
		return true;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Change the Clipboard Items' order
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	protected function ReorderClipboardItems($order=false){
 | 
						|
		global $langmessage;
 | 
						|
		$this->ajaxReplace = [];
 | 
						|
 | 
						|
		if( !is_array($order) ){
 | 
						|
			if( !is_array($_POST['order']) ){
 | 
						|
				msg($langmessage['OOPS'] . ' (Section Clipboard - Reorder Items: Invalid Request');
 | 
						|
				return false;
 | 
						|
			}
 | 
						|
			$order = $_POST['order'];
 | 
						|
		}
 | 
						|
 | 
						|
		$clipboard_data = \gp\tool\Files::GetSectionClipboard();
 | 
						|
 | 
						|
		if( count($order) != count($clipboard_data) ){
 | 
						|
			msg($langmessage['OOPS'] . ' (Section Clipboard - Reorder Items: Item count mismatch');
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
 | 
						|
		foreach( $order as $key => $index ){
 | 
						|
			$new_clipboard_data[$key] = $clipboard_data[$index];
 | 
						|
		}
 | 
						|
 | 
						|
		$clipboard_data = $new_clipboard_data;
 | 
						|
		unset($new_clipboard_data);
 | 
						|
 | 
						|
		if( \gp\tool\Files::SaveSectionClipboard($clipboard_data) ){
 | 
						|
			// msg($langmessage['SAVED']);
 | 
						|
		}else{
 | 
						|
			msg($langmessage['OOPS'] . ' (Section Clipboard - Reorder Items: Save data failed)');
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
 | 
						|
		$clipboard_links = self::SectionClipboardLinks($clipboard_data);
 | 
						|
		$this->ajaxReplace[] = ['inner', '#section-clipboard-items', $clipboard_links];
 | 
						|
		$this->ajaxReplace[] = ['loaded', '', ''];
 | 
						|
 | 
						|
		return true;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Change the label of a Clipboard Item
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	protected function RelabelClipboardItem($item_num=false,$new_label=false){
 | 
						|
		global $langmessage;
 | 
						|
		$this->ajaxReplace = [];
 | 
						|
 | 
						|
		if( !$item_num ){
 | 
						|
			if( !isset($_POST['item_number']) || !ctype_digit($_POST['item_number']) ){
 | 
						|
				msg($langmessage['OOPS'] . ' (Section Clipboard - Remove Item: Invalid Request');
 | 
						|
				return false;
 | 
						|
			}
 | 
						|
			$item_num = $_POST['item_number'];
 | 
						|
		}
 | 
						|
 | 
						|
		if( !$new_label ){
 | 
						|
			if( !isset($_POST['new_label']) ){
 | 
						|
				msg($langmessage['OOPS'] . ' (Section Clipboard - Relabel Item: Invalid Label');
 | 
						|
				return false;
 | 
						|
			}
 | 
						|
			$new_label = htmlspecialchars($_POST['new_label']);
 | 
						|
		}
 | 
						|
 | 
						|
		$clipboard_data = \gp\tool\Files::GetSectionClipboard();
 | 
						|
 | 
						|
		if( !isset($clipboard_data[$item_num]) ){
 | 
						|
			msg($langmessage['OOPS'] . ' (Section Clipboard - Relabel Item: Invalid Item Number (' . $item_num . ')');
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
 | 
						|
		$clipboard_data = array_values($clipboard_data);
 | 
						|
 | 
						|
		$clipboard_data[$item_num]['label'] = $new_label;
 | 
						|
 | 
						|
		if( \gp\tool\Files::SaveSectionClipboard($clipboard_data) ){
 | 
						|
			// msg($langmessage['SAVED']);
 | 
						|
		}else{
 | 
						|
			msg($langmessage['OOPS'] . ' (Section Clipboard - Remove Item: Save data failed)');
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
 | 
						|
		$clipboard_links = self::SectionClipboardLinks($clipboard_data);
 | 
						|
		$this->ajaxReplace[] = ['inner', '#section-clipboard-items', $clipboard_links];
 | 
						|
		$this->ajaxReplace[] = ['loaded', '', ''];
 | 
						|
 | 
						|
		return true;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Extract sections from the current page to be stored in the Clipboard
 | 
						|
	 * TS 5.1.1-b1: fix nesting error, changing from recursion to while iteration + counter
 | 
						|
	 */
 | 
						|
	public function ExtractSections($section_num){
 | 
						|
		$counter = 0;
 | 
						|
		$sections = [];
 | 
						|
		while( $counter >= 0 ){
 | 
						|
			$section_data = $this->file_sections[$section_num];
 | 
						|
 | 
						|
			if( $section_data['type'] == 'wrapper_section' && isset($section_data['contains_sections']) ){
 | 
						|
				$counter += $section_data['contains_sections'];
 | 
						|
			}
 | 
						|
 | 
						|
			// remove possible hidden state
 | 
						|
			$section_data['gp_hidden'] = false;
 | 
						|
 | 
						|
			// add section
 | 
						|
			$sections[] = $section_data;
 | 
						|
 | 
						|
			$section_num++;
 | 
						|
			$counter--;
 | 
						|
		}
 | 
						|
 | 
						|
		return $sections;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Get the Section Clipboard links
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public static function SectionClipboardLinks($clipboard_data=false){
 | 
						|
		global $langmessage;
 | 
						|
		$clipboard_data = $clipboard_data ? $clipboard_data : \gp\tool\Files::GetSectionClipboard();
 | 
						|
 | 
						|
		if( empty($clipboard_data) ){
 | 
						|
			return '<li class="clipboard-empty-msg"><i class="fa fa-clipboard"></i> ' . $langmessage['Clipboard'] . '</li>';
 | 
						|
		}
 | 
						|
 | 
						|
		$clipboard_links = '';
 | 
						|
 | 
						|
		foreach( $clipboard_data as $key => $clipboard_item ){
 | 
						|
			$icon_class = $clipboard_item['type'] == 'wrapper_section' ? 'fa fa-fw fa-clone' : 'fa fa-fw fa-square-o';
 | 
						|
			// $icon_title = $clipboard_item['label'];
 | 
						|
			// $content = \gp\tool\Output\Sections::GetSection($clipboard_item['file_sections'], 0); // private method :-(
 | 
						|
			$content = $clipboard_item['content'];
 | 
						|
			$response = htmlspecialchars($content);
 | 
						|
 | 
						|
			$clipboard_links .= '<li style="border-left:4px solid ' . $clipboard_item['color'] . ';" data-item_index="' . $key . '">';
 | 
						|
 | 
						|
			$clipboard_links .= '<a class="remove-clipboard-item" ';
 | 
						|
			$clipboard_links .= 	'title="' . $langmessage['remove'] . '" ';
 | 
						|
			$clipboard_links .= 	'data-cmd="RemoveSectionClipboardItem">';
 | 
						|
			$clipboard_links .= 	'<i class="fa fa-trash"></i>';
 | 
						|
			$clipboard_links .= '</a>';
 | 
						|
 | 
						|
			$clipboard_links .= '<a class="relabel-clipboard-item" ';
 | 
						|
			$clipboard_links .= 	'title="' . $langmessage['label'] . '" ';
 | 
						|
			$clipboard_links .= 	'data-cmd="RelabelSectionClipboardItem">';
 | 
						|
			$clipboard_links .= 	'<i class="fa fa-i-cursor"></i>'; // fa-pencil-square-o
 | 
						|
			$clipboard_links .= '</a>';
 | 
						|
 | 
						|
			$clipboard_links .= '<a class="preview_section" ';
 | 
						|
			$clipboard_links .= 	'title="' . $langmessage['add'] . ' (' . count($clipboard_item['file_sections']) . ')" ';
 | 
						|
			$clipboard_links .= 	'data-cmd="AddFromClipboard" ';
 | 
						|
			$clipboard_links .= 	'data-response="' . $response . '" >';
 | 
						|
			$clipboard_links .= 	'<i class="clipboard-item-icon ' . $icon_class . '"></i> ';
 | 
						|
			$clipboard_links .= 	'<i class="clipboard-item-label-wrap">';
 | 
						|
			$clipboard_links .= 		'<span class="clipboard-item-label">' . $clipboard_item['label'] . '</span>';
 | 
						|
			$clipboard_links .= 	'</i>';
 | 
						|
			$clipboard_links .= '</a>';
 | 
						|
 | 
						|
			$clipboard_links .= '</li>';
 | 
						|
		}
 | 
						|
 | 
						|
		return $clipboard_links;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Get the posted attributes for a section
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	protected function PostedAttributes(&$section, $i){
 | 
						|
		global $dirPrefix;
 | 
						|
 | 
						|
		if( isset($_POST['attributes'][$i]) && is_array($_POST['attributes'][$i]) ){
 | 
						|
			foreach($_POST['attributes'][$i] as $attr_name => $attr_value){
 | 
						|
 | 
						|
				$attr_name		= strtolower($attr_name);
 | 
						|
				$attr_name		= trim($attr_name);
 | 
						|
				$attr_value		= trim($attr_value);
 | 
						|
 | 
						|
				if( empty($attr_name) || empty($attr_value) || $attr_name == 'id' || substr($attr_name, 0, 7) == 'data-gp' ){
 | 
						|
					continue;
 | 
						|
				}
 | 
						|
 | 
						|
				//strip $dirPrefix
 | 
						|
				if( $attr_name == 'src' && !empty($dirPrefix) && strpos($attr_value,$dirPrefix) === 0 ){
 | 
						|
					$attr_value = substr($attr_value, strlen($dirPrefix));
 | 
						|
				}
 | 
						|
 | 
						|
				$section['attributes'][$attr_name] = $attr_value;
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Perform various section editing commands
 | 
						|
	 * @return bool
 | 
						|
	 */
 | 
						|
	public function SectionEdit(){
 | 
						|
		global $langmessage, $page;
 | 
						|
 | 
						|
		$page->ajaxReplace = [];
 | 
						|
 | 
						|
		$section_num = $_REQUEST['section'];
 | 
						|
		if( !is_numeric($section_num) || !isset($this->file_sections[$section_num])){
 | 
						|
			echo 'false;';
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
 | 
						|
		$cmd = \gp\tool::GetCommand();
 | 
						|
 | 
						|
		if( !\gp\tool\Editing::SectionEdit($cmd, $this->file_sections[$section_num], $section_num, $this->title, $this->file_stats) ){
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
 | 
						|
		//save if the file was changed
 | 
						|
		if( !$this->SaveThis() ){
 | 
						|
			msg($langmessage['OOPS'].' (SE3)');
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
 | 
						|
		$page->ajaxReplace[] = ['ck_saved', '', ''];
 | 
						|
 | 
						|
		//update gallery information
 | 
						|
		switch($this->file_sections[$section_num]['type']){
 | 
						|
			case 'gallery':
 | 
						|
				$this->GalleryEdited();
 | 
						|
			break;
 | 
						|
		}
 | 
						|
 | 
						|
		\gp\admin\Notifications::UpdateNotifications();
 | 
						|
 | 
						|
		return true;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Recalculate the file_type string for this file
 | 
						|
	 * Updates $this->meta_data and $gp_titles
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public function ResetFileTypes(){
 | 
						|
		global $gp_titles;
 | 
						|
 | 
						|
		$new_types = [];
 | 
						|
		foreach($this->file_sections as $section){
 | 
						|
			$new_types[] = $section['type'];
 | 
						|
		}
 | 
						|
		$new_types = array_unique($new_types);
 | 
						|
		$new_types = array_diff($new_types, ['']);
 | 
						|
		sort($new_types);
 | 
						|
 | 
						|
		$new_types = implode(',', $new_types);
 | 
						|
		$this->meta_data['file_type'] = $new_types;
 | 
						|
 | 
						|
		if( !isset($gp_titles[$this->gp_index]) ){
 | 
						|
			return;
 | 
						|
		}
 | 
						|
 | 
						|
		$gp_titles[$this->gp_index]['type'] = $new_types;
 | 
						|
		\gp\admin\Tools::SavePagesPHP();
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Return a list of section types
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public static function NewSections($checkboxes=false){
 | 
						|
		global $langmessage;
 | 
						|
 | 
						|
		$types_with_icons = ['text', 'image', 'gallery', 'wrapper_section', 'include'];
 | 
						|
 | 
						|
		$section_types = \gp\tool\Output\Sections::GetTypes();
 | 
						|
 | 
						|
		$links = [];
 | 
						|
 | 
						|
		foreach($section_types as $type => $type_info){
 | 
						|
			$img = '';
 | 
						|
			if( in_array($type, $types_with_icons) ){
 | 
						|
				$img = \gp\tool::GetDir('/include/imgs/section-' . $type . '.png');
 | 
						|
			}
 | 
						|
			$links[] = [$type, $img, '', $type_info['file_under']];
 | 
						|
		}
 | 
						|
 | 
						|
		//section combo: text & image
 | 
						|
		$links[] = [
 | 
						|
			['text.gpCol-6', 'image.gpCol-6'],
 | 
						|
			\gp\tool::GetDir('/include/imgs/section-combo-text-image.png'),
 | 
						|
			[
 | 
						|
				'gp_label'		=> 'Text & Image',
 | 
						|
				'gp_color'		=> '#555',
 | 
						|
				'attributes'	=> [
 | 
						|
					'class'		=> 'gpRow',
 | 
						|
				],
 | 
						|
			],
 | 
						|
			'default', // file_under, req as of TS 5.2
 | 
						|
		];
 | 
						|
 | 
						|
		//section combo: text & gallery
 | 
						|
		$links[] = [
 | 
						|
			['text.gpCol-6', 'gallery.gpCol-6'],
 | 
						|
			\gp\tool::GetDir('/include/imgs/section-combo-text-gallery.png'),
 | 
						|
			[
 | 
						|
				'gp_label'		=> 'Text & Gallery',
 | 
						|
				'gp_color'		=> '#555',
 | 
						|
				'attributes'	=> [
 | 
						|
					'class'		=> 'gpRow',
 | 
						|
				],
 | 
						|
			],
 | 
						|
			'default',
 | 
						|
		];
 | 
						|
 | 
						|
		//section combo: 3 text columns
 | 
						|
		$links[] = [
 | 
						|
			['text.gpCol-4', 'text.gpCol-4', 'text.gpCol-4'],
 | 
						|
			\gp\tool::GetDir('/include/imgs/section-combo-3-text-cols.png'),
 | 
						|
			[
 | 
						|
				'gp_label'		=> '3 Text Columns',
 | 
						|
				'gp_color'		=> '#555',
 | 
						|
				'attributes'	=> [
 | 
						|
					'class'		=> 'gpRow',
 | 
						|
				],
 | 
						|
			],
 | 
						|
			'default',
 | 
						|
		];
 | 
						|
 | 
						|
		/**
 | 
						|
		 * FYI when using the 'NewSections' plugin filter hook
 | 
						|
		 *
 | 
						|
		 * a new section subarray looks like this:
 | 
						|
		 *
 | 
						|
		 * array(
 | 
						|
		 * 	[0]	=>	(string)section type  OR  (array) of (string)section types (or subsequent wrapper arrays)
 | 
						|
		 * 				if we pass an array, the new section is a wrapper (combo) containing nested sections
 | 
						|
		 * 				while every (string)section type MAY contain one or multiple css classes
 | 
						|
		 * 					css classes are everything in (string)section type after a dot
 | 
						|
		 * 					multiple css classes are separated by space charaters
 | 
						|
		 *
 | 
						|
		 * 	[1]	=>	(string) relative path to an admin UI icon for that section type
 | 
						|
		 * 				if empty, no icon will appear
 | 
						|
		 *
 | 
						|
		 * 	[2]	=>	(boolean)false  OR  (string)wrapper class name(s)  OR  (array)wrapper data
 | 
						|
		 * 				when passing wrapper classes string, multiple class names are separated by space characters
 | 
						|
		 * 				when passing a wrapper data array, it may contain arbitrary section data key-pairs
 | 
						|
		 * 					including an array of html attributes like 'class'
 | 
						|
		 *
 | 
						|
		 * 	[3]	=>	as of Typesetter 5.2, optional (string)file_under
 | 
						|
		 * 				use 'hidden' if you do not want the link to appear in the admi UI
 | 
						|
		 * 				will become 'plugins' by default if not defined
 | 
						|
		 * )
 | 
						|
		 *
 | 
						|
		 * Furthermore: instead of existing 'real' section types, you can also use pseudo types
 | 
						|
		 * 	pseudo types must to be defined via the GetDefaultContent filter hook
 | 
						|
		 * 		where a section array must be returned, which has the real section type string in the 'type' key
 | 
						|
		 *
 | 
						|
		 * This is a grown structure and we apologize that it is so messy
 | 
						|
		 *
 | 
						|
		 * For handling more complex new sections and even multi-level-nested section combos see e.g.
 | 
						|
		 * 	the SliderFactory plugin (https://www.typesettercms.com/Plugins/310_Slider_Factory)
 | 
						|
		 *
 | 
						|
		 */
 | 
						|
		$links = \gp\tool\Plugins::Filter('NewSections', [$links]);
 | 
						|
 | 
						|
		$new_section_links = [];
 | 
						|
		foreach( $links as $link ){
 | 
						|
			$link += ['', '', false, 'plugins']; // $link[2] will be replaced in NewSectionLink() if missing
 | 
						|
			$type_id = substr(base_convert(md5(json_encode((array)$link[0])), 16, 32), 0, 6); // creates a unique id for every entry
 | 
						|
			$new_section_links[$type_id] = $link;
 | 
						|
		}
 | 
						|
 | 
						|
		// last chance to filter, re-arrange or modify new section links
 | 
						|
		// before they appear in the admin UI (based on their $type_id or other criteria)
 | 
						|
		$new_section_links = \gp\tool\Plugins::Filter('NewSectionLinks', [$new_section_links]);
 | 
						|
 | 
						|
		$stackers = [];
 | 
						|
 | 
						|
		foreach( $new_section_links as $type_id => $link ){
 | 
						|
			$file_under		= $link[3];
 | 
						|
			$stackers[$file_under][$type_id] = $link;
 | 
						|
		}
 | 
						|
 | 
						|
		// remove all section types that are filed under 'hidden'
 | 
						|
		unset($stackers['hidden']);
 | 
						|
 | 
						|
		// if there are more than 12 links and more than 1 remaining stackers, split the output into an accordion
 | 
						|
		$use_stackers = count($stackers) > 1 && $new_section_links > 12;
 | 
						|
 | 
						|
		$stacker_collapsed = false;
 | 
						|
		foreach( $stackers as $stacker => $links ){
 | 
						|
			if( $use_stackers ){
 | 
						|
				$stacker_label = isset($langmessage[$stacker]) ? $langmessage[$stacker] : htmlspecialchars($stacker);
 | 
						|
				echo '<section class="collapsible">';
 | 
						|
				echo 	'<h4 class="head'. ($stacker_collapsed ? ' gp_collapsed' : '') . '">';
 | 
						|
				echo 		'<a data-cmd="collapsible">' . $stacker_label . '</a>';
 | 
						|
				echo 	'</h4>';
 | 
						|
				echo 	'<div class="collapsearea' . ($stacker_collapsed ? ' nodisplay' : '') . '">';
 | 
						|
			}
 | 
						|
			foreach( $links as $link ){
 | 
						|
				$types			= $link[0];
 | 
						|
				$icon			= $link[1];
 | 
						|
				$wrapper_data	= $link[2];
 | 
						|
				$file_under		= $link[3];
 | 
						|
 | 
						|
				echo self::NewSectionLink($types, $icon, $wrapper_data, $checkboxes, $type_id, $file_under);
 | 
						|
			}
 | 
						|
			if( $use_stackers ){
 | 
						|
				echo 	'</div>';
 | 
						|
				echo '</section>';
 | 
						|
				$stacker_collapsed = true;
 | 
						|
			}
 | 
						|
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Add link to manage section admin for nested section type
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public static function NewSectionLink($types, $icon, $wrapper_data=false, $checkbox=false, $type_id='undefined', $file_under=''){
 | 
						|
		global $dataDir, $page, $langmessage;
 | 
						|
 | 
						|
		$types = (array)$types;
 | 
						|
 | 
						|
		$is_wrapper = count([$types]) > 1 || is_array($types[0]);
 | 
						|
 | 
						|
		if( $is_wrapper && !$wrapper_data ){
 | 
						|
			// add default wrapper data if undefined
 | 
						|
			$wrapper_data = [
 | 
						|
				'gp_label'		=> $langmessage['Section Wrapper'],
 | 
						|
				'gp_color'		=> '#555',
 | 
						|
				'attributes'	=> [
 | 
						|
					// 'class' => 'gpRow', // we don't use gpRow as default class for new wrappers anymore
 | 
						|
				],
 | 
						|
			];
 | 
						|
		}
 | 
						|
 | 
						|
		static $fi = 0;
 | 
						|
 | 
						|
		$text_label = $is_wrapper && isset($wrapper_data['gp_label']) ? $wrapper_data['gp_label'] : self::SectionLabel($types);
 | 
						|
 | 
						|
		$label = '<span>' . $text_label . '</span>';
 | 
						|
		if( !empty($icon) ){
 | 
						|
			$label .= '<img src="' . $icon . '"/>';
 | 
						|
		}
 | 
						|
 | 
						|
		//checkbox used for new pages
 | 
						|
		if( $checkbox ){
 | 
						|
 | 
						|
			if( count($types) > 1 || is_array($types[0]) ){ // == nested sections
 | 
						|
				$q = ['types' => $types, 'wrapper_data' => $wrapper_data];
 | 
						|
				$q = json_encode($q);
 | 
						|
			}else{
 | 
						|
				$q = $types[0];
 | 
						|
			}
 | 
						|
 | 
						|
			//checked
 | 
						|
			$checked = '';
 | 
						|
			if( isset($_REQUEST['content_type']) && $_REQUEST['content_type'] == $q ){
 | 
						|
				$checked = ' checked="checked"';
 | 
						|
			}elseif( empty($_REQUEST['content_type']) && $fi === 0 ){
 | 
						|
				$checked = ' checked="checked"';
 | 
						|
				$fi++;
 | 
						|
			}
 | 
						|
 | 
						|
			$id = 'checkbox_' . md5($q);
 | 
						|
			echo '<div class="new_section_link" data-type-id="' . $type_id . '" data-file-under="' . $file_under . '">';
 | 
						|
			echo   '<input name="content_type" type="radio" ';
 | 
						|
			echo     'value="' . htmlspecialchars($q) . '" id="' . $id . '" ';
 | 
						|
			echo     'required="required" ' . $checked . ' />';
 | 
						|
			echo   '<label title="' . $text_label . '" for="' . $id . '">';
 | 
						|
			echo     $label;
 | 
						|
			echo   '</label>';
 | 
						|
			echo '</div>';
 | 
						|
			return;
 | 
						|
		} // /if $checkboxes
 | 
						|
 | 
						|
		//links used for new sections
 | 
						|
		$attrs = [
 | 
						|
			'data-cmd' => 'AddSection',
 | 
						|
			'class' => 'preview_section',
 | 
						|
		];
 | 
						|
		if( count($types) > 1 || is_array($types[0]) ){
 | 
						|
			$attrs['data-response'] = $page->NewNestedSection($types, $wrapper_data);
 | 
						|
		}else{
 | 
						|
			$attrs['data-response'] = $page->GetNewSection($types[0]);
 | 
						|
		}
 | 
						|
 | 
						|
		$return =  '<div class="new_section_link" data-type-id="' . $type_id . '" data-file-under="' . $file_under . '">';
 | 
						|
		$return .=   '<a ' . \gp\tool::LinkAttr($attrs, $label) . '>' . $label . '</a>';
 | 
						|
		$return .= '</div>';
 | 
						|
 | 
						|
		return $return;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Return a readable label for the section
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public static function SectionLabel($types){
 | 
						|
		$section_types	= \gp\tool\Output\Sections::GetTypes();
 | 
						|
		$text_label		= [];
 | 
						|
 | 
						|
		foreach($types as $type){
 | 
						|
 | 
						|
			if( is_array($type) ){
 | 
						|
				continue;
 | 
						|
			}
 | 
						|
			self::TypeClass($type);
 | 
						|
 | 
						|
			if( isset($section_types[$type]) ){
 | 
						|
				$text_label[] = $section_types[$type]['label'];
 | 
						|
			}else{
 | 
						|
				$text_label[] = $type;
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		return implode(' & ', $text_label);
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Split the type and class from $type = div.classname into $type = div, $class = classname
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public static function TypeClass(&$type){
 | 
						|
 | 
						|
		$class = '';
 | 
						|
 | 
						|
		if( !is_array($type) && strpos($type, '.') ){
 | 
						|
			list($type,$class) = explode('.', $type, 2);
 | 
						|
		}
 | 
						|
 | 
						|
		return $class;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Save the current page
 | 
						|
	 * Save a backup if $backup is true
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public function SaveThis($backup=true){
 | 
						|
 | 
						|
		//return true if nothing has changed
 | 
						|
		if( $backup && $this->checksum === $this->Checksum() ){
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
 | 
						|
		//file count
 | 
						|
		if( !isset($this->meta_data['file_number']) ){
 | 
						|
			$this->meta_data['file_number'] = \gp\tool\Files::NewFileNumber();
 | 
						|
		}
 | 
						|
 | 
						|
		if( $backup ){
 | 
						|
			$this->SaveBackup(); //make a backup of the page file
 | 
						|
		}
 | 
						|
 | 
						|
		if( isset($_POST['prevent_draft']) && !$this->draft_exists ){
 | 
						|
			if( !\gp\tool\Files::SaveData($this->file, 'file_sections', $this->file_sections, $this->meta_data) ){
 | 
						|
				return false;
 | 
						|
			}
 | 
						|
		}else{
 | 
						|
			if( !\gp\tool\Files::SaveData($this->draft_file, 'file_sections', $this->file_sections, $this->meta_data) ){
 | 
						|
				return false;
 | 
						|
			}
 | 
						|
			$this->draft_exists = true;
 | 
						|
		}
 | 
						|
 | 
						|
		// update notifications
 | 
						|
		\gp\admin\Notifications::UpdateNotifications();
 | 
						|
 | 
						|
		return true;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Generate a checksum for this page, used to determine if the page content has been edited
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public function Checksum(){
 | 
						|
		$temp = [];
 | 
						|
		foreach($this->file_sections as $section){
 | 
						|
			unset($section['modified'], $section['modified_by']);
 | 
						|
			$temp[] = $section;
 | 
						|
		}
 | 
						|
		$checksum = serialize($temp);
 | 
						|
		return sha1($checksum) . md5($checksum);
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 *	Save a backup of the file
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public function SaveBackup(){
 | 
						|
		global $dataDir, $gpAdmin;
 | 
						|
 | 
						|
		$dir	= $dataDir . '/data/_backup/pages/' . $this->gp_index;
 | 
						|
		$time	= \gp\tool\Editing::ReqTime(); //use the request time
 | 
						|
 | 
						|
		//just one backup per edit session (auto-saving would create too many backups otherwise)
 | 
						|
		$previous_backup	= $this->BackupFile($time);
 | 
						|
		if( !empty($previous_backup) ){
 | 
						|
			return true;
 | 
						|
		}
 | 
						|
 | 
						|
		// get the raw contents so we can add the size to the backup file name
 | 
						|
		if( $this->draft_exists ){
 | 
						|
			$contents	= \gp\tool\Files::GetRaw($this->draft_file);
 | 
						|
		}else{
 | 
						|
			$contents	= \gp\tool\Files::GetRaw($this->file);
 | 
						|
		}
 | 
						|
 | 
						|
		//backup file name
 | 
						|
		$len				= strlen($contents);
 | 
						|
		$backup_file		= $dir . '/' . $time . '.' . $len . '.' . $gpAdmin['username'];
 | 
						|
 | 
						|
		//compress
 | 
						|
		if( function_exists('gzencode') && function_exists('readgzfile') ){
 | 
						|
			$backup_file .= '.gze';
 | 
						|
			$contents = gzencode($contents,9);
 | 
						|
		}
 | 
						|
 | 
						|
		if( !\gp\tool\Files::Save($backup_file, $contents) ){
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
 | 
						|
		$this->CleanBackupFolder();
 | 
						|
		return true;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Reduce the number of files in the backup folder
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public function CleanBackupFolder(){
 | 
						|
		global $dataDir, $config;
 | 
						|
 | 
						|
		$files			= $this->BackupFiles();
 | 
						|
		$file_count		= count($files);
 | 
						|
 | 
						|
		if( $file_count <= $config['history_limit'] ){
 | 
						|
			return;
 | 
						|
		}
 | 
						|
		$delete_count = $file_count - $config['history_limit'];
 | 
						|
		$files = array_splice($files, 0, $delete_count);
 | 
						|
		foreach($files as $file){
 | 
						|
			$full_path = $dataDir . '/data/_backup/pages/' . $this->gp_index . '/' . $file;
 | 
						|
			unlink($full_path);
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Make the working draft the live file
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public function PublishDraft(){
 | 
						|
		global $langmessage, $page;
 | 
						|
 | 
						|
		if( !$this->draft_exists ){
 | 
						|
			msg($langmessage['OOPS'] . ' (Not a draft)');
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
 | 
						|
		if( !\gp\tool\Files::SaveData($this->file, 'file_sections', $this->file_sections, $this->meta_data) ){
 | 
						|
			msg($langmessage['OOPS'] . ' (Draft not published)');
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
 | 
						|
		$draft_file = \gp\tool\Files::FilePath($this->draft_file);
 | 
						|
		unlink($draft_file);
 | 
						|
		$this->ResetFileTypes();
 | 
						|
		$this->draft_exists = false;
 | 
						|
 | 
						|
		$page->ajaxReplace		= [];
 | 
						|
		$page->ajaxReplace[]	= ['DraftPublished'];
 | 
						|
 | 
						|
		\gp\admin\Notifications::UpdateNotifications();
 | 
						|
 | 
						|
		return true;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Display the contents of a past revision
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	protected function ViewRevision(){
 | 
						|
 | 
						|
		\gp\admin\Tools::$show_toolbar	= false;
 | 
						|
		$revision						=& $_REQUEST['revision'];
 | 
						|
		$file_sections					= $this->GetRevision($revision);
 | 
						|
 | 
						|
		$this->head_js[]				= '/include/js/admin/revision.js';
 | 
						|
 | 
						|
		if( $file_sections === false ){
 | 
						|
			$this->DefaultDisplay();
 | 
						|
			return;
 | 
						|
		}
 | 
						|
 | 
						|
 | 
						|
		echo \gp\tool\Output\Sections::Render($file_sections, $this->title, \gp\tool\Files::$last_stats);
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * View the current public facing version of the file
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public function ViewCurrent(){
 | 
						|
 | 
						|
		\gp\admin\Tools::$show_toolbar		= false;
 | 
						|
		$this->head_js[]					= '/include/js/admin/revision.js';
 | 
						|
 | 
						|
		if( !$this->draft_exists ){
 | 
						|
			$this->DefaultDisplay();
 | 
						|
			return;
 | 
						|
		}
 | 
						|
 | 
						|
		$file_sections			= \gp\tool\Files::Get($this->file, 'file_sections');
 | 
						|
		echo \gp\tool\Output\Sections::Render($file_sections, $this->title, $this->file_stats);
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Get the contents of a revision
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	protected function GetRevision($time){
 | 
						|
 | 
						|
		$full_path	= $this->BackupFile($time);
 | 
						|
 | 
						|
		if( is_null($full_path) ){
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
 | 
						|
		//if it's a compressed file, we need an uncompressed version
 | 
						|
		if( strpos($full_path, '.gze') !== false ){
 | 
						|
 | 
						|
			ob_start();
 | 
						|
			readgzfile($full_path);
 | 
						|
			$contents		= ob_get_clean();
 | 
						|
 | 
						|
			$full_path		= substr($full_path, 0, -3) . 'php';
 | 
						|
			$full_path		= \gp\tool\Files::FilePath($full_path);
 | 
						|
			\gp\tool\Files::Save($full_path, $contents);
 | 
						|
			$file_sections	= \gp\tool\Files::Get($full_path, 'file_sections');
 | 
						|
			unlink($full_path);
 | 
						|
 | 
						|
		}else{
 | 
						|
			$file_sections	= \gp\tool\Files::Get($full_path, 'file_sections');
 | 
						|
		}
 | 
						|
 | 
						|
		return $file_sections;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Return a list of the available backup for the current file
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public function BackupFiles(){
 | 
						|
		global $dataDir;
 | 
						|
		$dir = $dataDir . '/data/_backup/pages/' . $this->gp_index;
 | 
						|
 | 
						|
		if( !file_exists($dir) ){
 | 
						|
			return [];
 | 
						|
		}
 | 
						|
		$all_files = scandir($dir);
 | 
						|
		$files = [];
 | 
						|
		foreach($all_files as $file){
 | 
						|
			if( $file == '.' || $file == '..' ){
 | 
						|
				continue;
 | 
						|
			}
 | 
						|
			$parts = explode('.', $file);
 | 
						|
			$time = array_shift($parts);
 | 
						|
			if( !is_numeric($time) ){
 | 
						|
				continue;
 | 
						|
			}
 | 
						|
			$files[$time] = $file;
 | 
						|
		}
 | 
						|
 | 
						|
		ksort($files);
 | 
						|
		return $files;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Return the full path of the saved revision if it exists
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public function BackupFile( $time ){
 | 
						|
		global $dataDir;
 | 
						|
 | 
						|
		$files = $this->BackupFiles();
 | 
						|
		if( !isset($files[$time]) ){
 | 
						|
			return;
 | 
						|
		}
 | 
						|
 | 
						|
		return $dataDir . '/data/_backup/pages/' . $this->gp_index . '/' . $files[$time];
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Extract information about the gallery from it's html: img_count, icon_src
 | 
						|
	 * Call GalleryEdited when a gallery section is removed, edited
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public function GalleryEdited(){
 | 
						|
		\gp\special\Galleries::UpdateGalleryInfo($this->title, $this->file_sections);
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	public function GetSection(&$section_num){
 | 
						|
		global $langmessage;
 | 
						|
 | 
						|
		if( !isset($this->file_sections[$section_num]) ){
 | 
						|
			trigger_error('invalid section number');
 | 
						|
			return;
 | 
						|
		}
 | 
						|
 | 
						|
		$curr_section_num								= $section_num;
 | 
						|
		$section_num++;
 | 
						|
 | 
						|
		$content										= '';
 | 
						|
		$section_data									= $this->file_sections[$curr_section_num];
 | 
						|
 | 
						|
		//make sure section_data is an array
 | 
						|
		$type											= gettype($section_data);
 | 
						|
		if( $type !== 'array' ){
 | 
						|
			trigger_error('$section_data is ' . $type . '. Array expected');
 | 
						|
			return;
 | 
						|
		}
 | 
						|
 | 
						|
		$section_data									+= ['attributes' => [], 'type' => 'text'];
 | 
						|
		$section_data['attributes']						+= ['class' => ''];
 | 
						|
		$orig_attrs										= $section_data['attributes'];
 | 
						|
		$section_data['attributes']['data-gp-section']	= $curr_section_num;
 | 
						|
		$section_types									= \gp\tool\Output\Sections::GetTypes();
 | 
						|
 | 
						|
		if( \gp\tool\Output::ShowEditLink() && \gp\admin\Tools::CanEdit($this->gp_index) ){
 | 
						|
 | 
						|
			if( isset($section_types[$section_data['type']]) ){
 | 
						|
				$title_attr		= $section_types[$section_data['type']]['label'];
 | 
						|
			}else{
 | 
						|
				$title_attr		= sprintf($langmessage['Section %s'], $curr_section_num+1);
 | 
						|
			}
 | 
						|
 | 
						|
			$attrs	= [
 | 
						|
						'title'		=> $title_attr,
 | 
						|
						'data-cmd'	=> 'inline_edit_generic',
 | 
						|
						'data-arg'	=> $section_data['type'] . '_inline_edit',
 | 
						|
					];
 | 
						|
			$link	= \gp\tool\Output::EditAreaLink(
 | 
						|
						$edit_index,
 | 
						|
						$this->title,
 | 
						|
						$langmessage['edit'],
 | 
						|
						'section='.$curr_section_num,$attrs
 | 
						|
					);
 | 
						|
 | 
						|
			$section_data['attributes']['data-gp-area-id']		= $edit_index;
 | 
						|
 | 
						|
			//included page target
 | 
						|
			$include_link = self::IncludeLink($section_data);
 | 
						|
 | 
						|
 | 
						|
			//section control links
 | 
						|
			if( $section_data['type'] != 'wrapper_section' ){
 | 
						|
				ob_start();
 | 
						|
				echo '<span class="nodisplay" id="ExtraEditLnks' . $edit_index . '">';
 | 
						|
				echo $link;
 | 
						|
				echo $include_link;
 | 
						|
				echo \gp\tool::Link(
 | 
						|
						$this->title,
 | 
						|
						$langmessage['Manage Sections'],
 | 
						|
						'cmd=ManageSections',
 | 
						|
						[
 | 
						|
							'class'		=> 'manage_sections',
 | 
						|
							'data-cmd'	=> 'inline_edit_generic',
 | 
						|
							'data-arg'	=> 'manage_sections',
 | 
						|
						]
 | 
						|
					);
 | 
						|
				echo '<span class="gp_separator"></span>';
 | 
						|
				echo \gp\tool::Link(
 | 
						|
						$this->title,
 | 
						|
						$langmessage['rename/details'],
 | 
						|
						'cmd=renameform&index=' . urlencode($this->gp_index),
 | 
						|
						['data-cmd' => 'gpajax']
 | 
						|
					);
 | 
						|
				echo \gp\tool::Link(
 | 
						|
						'/Admin/Revisions/'.$this->gp_index,
 | 
						|
						$langmessage['Revision History'],
 | 
						|
						''
 | 
						|
					);
 | 
						|
				echo '<span class="gp_separator"></span>';
 | 
						|
				echo \gp\tool::Link('Admin/Menu', $langmessage['file_manager']);
 | 
						|
				echo '</span>';
 | 
						|
				\gp\tool\Output::$editlinks .= ob_get_clean();
 | 
						|
			}
 | 
						|
 | 
						|
			$section_data['attributes']['id'] = 'ExtraEditArea' . $edit_index;
 | 
						|
		}
 | 
						|
 | 
						|
		$content 			.= $this->SectionNode($section_data, $orig_attrs);
 | 
						|
 | 
						|
		if( $section_data['type'] == 'wrapper_section' ){
 | 
						|
 | 
						|
			for( $cc=0; $cc < $section_data['contains_sections']; $cc++ ){
 | 
						|
				$content	.= $this->GetSection($section_num);
 | 
						|
			}
 | 
						|
 | 
						|
		}else{
 | 
						|
			\gp\tool\Output::$nested_edit 	= true;
 | 
						|
			$content .= \gp\tool\Output\Sections::RenderSection($section_data, $curr_section_num, $this->title, $this->file_stats);
 | 
						|
			\gp\tool\Output::$nested_edit 	= false;
 | 
						|
		}
 | 
						|
 | 
						|
		if( !isset($section_data['nodeName']) ){
 | 
						|
			$content		.= '<div class="gpclear"></div>';
 | 
						|
			$content		.= '</div>';
 | 
						|
		}else{
 | 
						|
			$content		.= \gp\tool\Output\Sections::EndTag($section_data['nodeName']);
 | 
						|
		}
 | 
						|
 | 
						|
		return $content;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Return a link to the included page or extra area
 | 
						|
	 *
 | 
						|
	 * @return string
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public static function IncludeLink($section_data){
 | 
						|
		global $langmessage;
 | 
						|
 | 
						|
		if( $section_data['type'] != 'include' || !array_key_exists('include_type',$section_data) ){
 | 
						|
			return '';
 | 
						|
		}
 | 
						|
 | 
						|
		if( isset($section_data['index']) ){
 | 
						|
			return \gp\tool::Link($section_data['content'], $langmessage['view/edit_page']);
 | 
						|
		}
 | 
						|
 | 
						|
		if( $section_data['include_type'] == 'extra' ){
 | 
						|
 | 
						|
			// get the real section type of the included extra area
 | 
						|
			$extra_sections		= \gp\tool\Output\Extra::ExtraContent($section_data['content']);
 | 
						|
			$extra_section_type	= $extra_sections[0]['type'];
 | 
						|
			if($extra_section_type != 'text' ){
 | 
						|
				// we can only edit text sections on Admin/Extra
 | 
						|
				return '';
 | 
						|
			}
 | 
						|
 | 
						|
			return \gp\tool::Link(
 | 
						|
				'Admin/Extra/' . rawurlencode($section_data['content']),
 | 
						|
				$langmessage['edit'] . ' » ' . str_replace('_', ' ', htmlspecialchars($section_data['content'])), // $langmessage['theme_content']
 | 
						|
				'cmd=EditExtra'
 | 
						|
			);
 | 
						|
		}
 | 
						|
 | 
						|
		return '';
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	public function GetSectionForClipboard(&$section_num){
 | 
						|
		global $langmessage;
 | 
						|
 | 
						|
		if( !isset($this->file_sections[$section_num]) ){
 | 
						|
			trigger_error('invalid section number');
 | 
						|
			return;
 | 
						|
		}
 | 
						|
 | 
						|
		$curr_section_num								= $section_num;
 | 
						|
		$section_num++;
 | 
						|
 | 
						|
		$content										= '';
 | 
						|
		$section_data									= $this->file_sections[$curr_section_num];
 | 
						|
 | 
						|
		//make sure section_data is an array
 | 
						|
		$type											= gettype($section_data);
 | 
						|
		if( $type !== 'array' ){
 | 
						|
			trigger_error('$section_data is ' . $type . '. Array expected');
 | 
						|
			return;
 | 
						|
		}
 | 
						|
 | 
						|
		$section_data									+= ['attributes' => [], 'type' => 'text'];
 | 
						|
		$section_data['attributes']						+= ['class' => ''];
 | 
						|
		// $section_data['attributes']['gp_type'] 			= $section_data['type'];
 | 
						|
		$section_data['gp_hidden']						= false;
 | 
						|
		$orig_attrs										= $section_data['attributes'];
 | 
						|
 | 
						|
		$content 			.= $this->SectionNode($section_data, $orig_attrs);
 | 
						|
 | 
						|
		if( $section_data['type'] == 'wrapper_section' ){
 | 
						|
 | 
						|
			for( $cc=0; $cc < $section_data['contains_sections']; $cc++ ){
 | 
						|
				$content	.= $this->GetSectionForClipboard($section_num);
 | 
						|
			}
 | 
						|
 | 
						|
		}else{
 | 
						|
			\gp\tool\Output::$nested_edit 	= true;
 | 
						|
			$content .= \gp\tool\Output\Sections::RenderSection($section_data, $curr_section_num, $this->title, $this->file_stats);
 | 
						|
			\gp\tool\Output::$nested_edit 	= false;
 | 
						|
		}
 | 
						|
 | 
						|
		if( !isset($section_data['nodeName']) ){
 | 
						|
			$content		.= '</div>';
 | 
						|
		}else{
 | 
						|
			$content		.= \gp\tool\Output\Sections::EndTag($section_data['nodeName']);
 | 
						|
		}
 | 
						|
 | 
						|
		return $content;
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	public function GalleryImages(){
 | 
						|
 | 
						|
		if( isset($_GET['dir']) ){
 | 
						|
			$dir_piece = $_GET['dir'];
 | 
						|
		}elseif( isset($this->meta_data['gallery_dir']) ){
 | 
						|
			$dir_piece = $this->meta_data['gallery_dir'];
 | 
						|
		}else{
 | 
						|
			$dir_piece = '/image';
 | 
						|
		}
 | 
						|
 | 
						|
		//remember browse directory
 | 
						|
		$this->meta_data['gallery_dir'] = $dir_piece;
 | 
						|
		$this->SaveThis(false);
 | 
						|
 | 
						|
		\gp\admin\Content\Uploaded::InlineList($dir_piece);
 | 
						|
	}
 | 
						|
 | 
						|
 | 
						|
	/**
 | 
						|
	 * Used by slideshow addons
 | 
						|
	 * @deprecated 3.6rc4
 | 
						|
	 *
 | 
						|
	 */
 | 
						|
	public function SaveSection_Text($section){
 | 
						|
		return \gp\tool\Editing::SectionFromPost_Text($this->file_sections[$section]);
 | 
						|
	}
 | 
						|
 | 
						|
}
 |