aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--Zotlabs/Module/Webpages.php222
-rw-r--r--include/text.php28
-rw-r--r--include/widgets.php14
-rw-r--r--view/pdl/mod_webpages.pdl1
-rw-r--r--view/theme/redbasic/css/style.css4
-rw-r--r--view/tpl/website_import_tools.tpl23
-rw-r--r--view/tpl/wiki.tpl3
7 files changed, 295 insertions, 0 deletions
diff --git a/Zotlabs/Module/Webpages.php b/Zotlabs/Module/Webpages.php
index cc0a01cce..9ece989a0 100644
--- a/Zotlabs/Module/Webpages.php
+++ b/Zotlabs/Module/Webpages.php
@@ -209,4 +209,226 @@ class Webpages extends \Zotlabs\Web\Controller {
return $o;
}
+ function post() {
+
+ if(($_FILES) && array_key_exists('zip_file',$_FILES)) {
+ $source = $_FILES["zip_file"]["tmp_name"];
+ $type = $_FILES["zip_file"]["type"];
+ $okay = false;
+ $accepted_types = array('application/zip', 'application/x-zip-compressed', 'multipart/x-zip', 'application/x-compressed');
+ foreach ($accepted_types as $mime_type) {
+ if ($mime_type == $type) {
+ $okay = true;
+ break;
+ }
+ }
+ if(!$okay) {
+ json_return_and_die(array('message' => 'Invalid file MIME type'));
+ }
+ $zip = new \ZipArchive();
+ if ($zip->open($source) === true) {
+ $tmp_folder_name = random_string(5);
+ $website = dirname($source) . '/' . $tmp_folder_name;
+ $zip->extractTo($website); // change this to the correct site path
+ $zip->close();
+ @unlink($source);
+
+ $hubsites = $this->import_website($website);
+ rrmdir($website);
+ $channel = \App::get_channel();
+ $blocks = $this->import_blocks($channel, $hubsites['blocks']);
+ logger('blocks imported: ' . json_encode($blocks));
+ }
+
+
+ }
+ }
+
+ private function import_website($path) {
+ $hubsites = [];
+ $pages = [];
+ $blocks = [];
+ $layouts = [];
+ // Import pages
+ $dirtoscan = $path . '/pages/';
+ if (is_dir($dirtoscan)) {
+ $dirlist = scandir($dirtoscan);
+ if ($dirlist) {
+ foreach ($dirlist as $element) {
+ if ($element === '.' || $element === '..') {
+ continue;
+ }
+ $folder = $dirtoscan . '/' . $element;
+ if (is_dir($folder)) {
+ $jsonfilepath = $folder . '/page.json';
+ if (is_file($jsonfilepath)) {
+ $pagejson = json_decode(file_get_contents($jsonfilepath), true);
+ $pagejson['path'] = $folder . '/' . $pagejson['contentfile'];
+ if ($pagejson['contentfile'] === '') {
+ logger('hubsites plugin: Invalid page content file');
+ return false;
+ }
+ $pagecontent = file_get_contents($folder . '/' . $pagejson['contentfile']);
+ if (!$pagecontent) {
+ logger('hubsites plugin: Failed to get file content for ' . $pagejson['contentfile']);
+ return false;
+ }
+ $pages[] = $pagejson;
+ }
+ }
+ }
+ }
+ }
+ $hubsites['pages'] = $pages;
+ // Import layouts
+ $dirtoscan = $path . '/layouts/';
+ if (is_dir($dirtoscan)) {
+ $dirlist = scandir($dirtoscan);
+ if ($dirlist) {
+ foreach ($dirlist as $element) {
+ if ($element === '.' || $element === '..') {
+ continue;
+ }
+ $folder = $dirtoscan . '/' . $element;
+ if (is_dir($folder)) {
+ $jsonfilepath = $folder . '/layout.json';
+ if (is_file($jsonfilepath)) {
+ $layoutjson = json_decode(file_get_contents($jsonfilepath), true);
+ $layoutjson['path'] = $folder . '/' . $layoutjson['contentfile'];
+ if ($layoutjson['contentfile'] === '') {
+ logger('hubsites plugin: Invalid layout content file');
+ return false;
+ }
+ $layoutcontent = file_get_contents($folder . '/' . $layoutjson['contentfile']);
+ if (!$layoutcontent) {
+ logger('hubsites plugin: Failed to get file content for ' . $layoutjson['contentfile']);
+ return false;
+ }
+ $layouts[] = $layoutjson;
+ }
+ }
+ }
+ }
+ }
+ $hubsites['layouts'] = $layouts;
+ // Import blocks
+ $dirtoscan = $path . '/blocks/';
+ if (is_dir($dirtoscan)) {
+ $dirlist = scandir($dirtoscan);
+ if ($dirlist) {
+ foreach ($dirlist as $element) {
+ if ($element === '.' || $element === '..') {
+ continue;
+ }
+ $folder = $dirtoscan . '/' . $element;
+ if (is_dir($folder)) {
+ $jsonfilepath = $folder . '/block.json';
+ if (is_file($jsonfilepath)) {
+ $block = json_decode(file_get_contents($jsonfilepath), true);
+ $block['path'] = $folder . '/' . $block['contentfile'];
+ if ($block['contentfile'] === '') {
+ logger('hubsites plugin: Invalid block content file');
+ return false;
+ }
+ $blockcontent = file_get_contents($folder . '/' . $block['contentfile']);
+ if (!$blockcontent) {
+ logger('hubsites plugin: Failed to get file content for ' . $block['contentfile']);
+ return false;
+ }
+ $blocks[] = $block;
+ }
+ }
+ }
+ }
+ }
+ $hubsites['blocks'] = $blocks;
+ //logger('hubsites: ' . json_encode($hubsites));
+ return $hubsites;
+ }
+
+ private function import_blocks($channel, $blocks) {
+ foreach ($blocks as &$b) {
+
+ $arr = array();
+ $arr['item_type'] = ITEM_TYPE_BLOCK;
+ $namespace = 'BUILDBLOCK';
+ $arr['uid'] = $channel['channel_id'];
+ $arr['aid'] = $channel['channel_account_id'];
+
+ $iid = q("select iid from item_id where service = 'BUILDBLOCK' and sid = '%s' and uid = %d",
+ dbesc($b['name']),
+ intval($channel['channel_id'])
+ );
+ if($iid) {
+ $iteminfo = q("select mid,created,edited from item where id = %d",
+ intval($iid[0]['iid'])
+ );
+ $arr['mid'] = $arr['parent_mid'] = $iteminfo[0]['mid'];
+ $arr['created'] = $iteminfo[0]['created'];
+ $arr['edited'] = (($b['edited']) ? datetime_convert('UTC', 'UTC', $b['edited']) : datetime_convert());
+ } else {
+ $arr['created'] = (($b['created']) ? datetime_convert('UTC', 'UTC', $b['created']) : datetime_convert());
+ $arr['edited'] = datetime_convert('UTC', 'UTC', '0000-00-00 00:00:00');
+ $arr['mid'] = $arr['parent_mid'] = item_message_id();
+ }
+ $arr['title'] = $b['title'];
+ $arr['body'] = file_get_contents($b['path']);
+ $arr['owner_xchan'] = get_observer_hash();
+ $arr['author_xchan'] = (($b['author_xchan']) ? $b['author_xchan'] : get_observer_hash());
+ if(($b['mimetype'] === 'text/bbcode' || $b['mimetype'] === 'text/html' ||
+ $b['mimetype'] === 'text/markdown' ||$b['mimetype'] === 'text/plain' ||
+ $b['mimetype'] === 'application/x-pdl' ||$b['mimetype'] === 'application/x-php')) {
+ $arr['mimetype'] = $b['mimetype'];
+ } else {
+ $arr['mimetype'] = 'text/bbcode';
+ }
+
+ $pagetitle = $b['name'];
+
+ // Verify ability to use html or php!!!
+ $execflag = false;
+ if ($arr['mimetype'] === 'application/x-php') {
+ $z = q("select account_id, account_roles, channel_pageflags from account left join channel on channel_account_id = account_id where channel_id = %d limit 1", intval(local_channel())
+ );
+
+ if ($z && (($z[0]['account_roles'] & ACCOUNT_ROLE_ALLOWCODE) || ($z[0]['channel_pageflags'] & PAGE_ALLOWCODE))) {
+ $execflag = true;
+ }
+ }
+
+ $remote_id = 0;
+
+ $z = q("select * from item_id where sid = '%s' and service = '%s' and uid = %d limit 1", dbesc($pagetitle), dbesc($namespace), intval(local_channel())
+ );
+
+ $i = q("select id, edited, item_deleted from item where mid = '%s' and uid = %d limit 1", dbesc($arr['mid']), intval(local_channel())
+ );
+ if ($z && $i) {
+ $remote_id = $z[0]['id'];
+ $arr['id'] = $i[0]['id'];
+ // don't update if it has the same timestamp as the original
+ if ($arr['edited'] > $i[0]['edited'])
+ $x = item_store_update($arr, $execflag);
+ } else {
+ if (($i) && (intval($i[0]['item_deleted']))) {
+ // was partially deleted already, finish it off
+ q("delete from item where mid = '%s' and uid = %d", dbesc($arr['mid']), intval(local_channel())
+ );
+ }
+ $x = item_store($arr, $execflag);
+ }
+ if ($x['success']) {
+ $item_id = $x['item_id'];
+ update_remote_id($channel, $item_id, $arr['item_type'], $pagetitle, $namespace, $remote_id, $arr['mid']);
+ $b['import_success'] = 1;
+ } else {
+ $b['import_success'] = 0;
+ }
+ }
+ return $blocks;
+}
+
+
+
+
}
diff --git a/include/text.php b/include/text.php
index 986e3b56c..096c8a5aa 100644
--- a/include/text.php
+++ b/include/text.php
@@ -2246,6 +2246,34 @@ function design_tools() {
));
}
+/**
+ * @brief Creates website import tools menu
+ *
+ * @return string
+ */
+function website_import_tools() {
+
+ $channel = App::get_channel();
+ $sys = false;
+
+ if(App::$is_sys && is_site_admin()) {
+ require_once('include/channel.php');
+ $channel = get_sys_channel();
+ $sys = true;
+ }
+
+ $who = $channel['channel_address'];
+
+ return replace_macros(get_markup_template('website_import_tools.tpl'), array(
+ '$title' => t('Import'),
+ //'$who' => $who,
+ '$import_label' => t('Import website...'),
+ '$import_placeholder' => t('Select folder to import'),
+ '$file_upload_text' => t('Import from a zipped folder:'),
+ '$file_import_text' => t('Import from cloud files:')
+ ));
+}
+
/* case insensitive in_array() */
function in_arrayi($needle, $haystack) {
diff --git a/include/widgets.php b/include/widgets.php
index 2d4d5b799..352c844ee 100644
--- a/include/widgets.php
+++ b/include/widgets.php
@@ -770,6 +770,20 @@ function widget_design_tools($arr) {
return design_tools();
}
+function widget_website_import_tools($arr) {
+
+ // mod menu doesn't load a profile. For any modules which load a profile, check it.
+ // otherwise local_channel() is sufficient for permissions.
+
+ if(App::$profile['profile_uid'])
+ if((App::$profile['profile_uid'] != local_channel()) && (! App::$is_sys))
+ return '';
+
+ if(! local_channel())
+ return '';
+
+ return website_import_tools();
+}
function widget_findpeople($arr) {
return findpeople_widget();
diff --git a/view/pdl/mod_webpages.pdl b/view/pdl/mod_webpages.pdl
index cef69f194..b62ec6e7c 100644
--- a/view/pdl/mod_webpages.pdl
+++ b/view/pdl/mod_webpages.pdl
@@ -1,3 +1,4 @@
[region=aside]
[widget=design_tools][/widget]
+[widget=website_import_tools][/widget]
[/region] \ No newline at end of file
diff --git a/view/theme/redbasic/css/style.css b/view/theme/redbasic/css/style.css
index 5768ccbd9..a63042b87 100644
--- a/view/theme/redbasic/css/style.css
+++ b/view/theme/redbasic/css/style.css
@@ -2039,3 +2039,7 @@ dl.bb-dl > dd > li {
border-style: solid;
border-width: 5px;
}
+
+#wiki-preview img {
+ max-width: 100%;
+} \ No newline at end of file
diff --git a/view/tpl/website_import_tools.tpl b/view/tpl/website_import_tools.tpl
new file mode 100644
index 000000000..ca9cc8b33
--- /dev/null
+++ b/view/tpl/website_import_tools.tpl
@@ -0,0 +1,23 @@
+<div id="website-import-tools" class="widget">
+ <h3>{{$title}}</h3>
+ <ul class="nav nav-pills nav-stacked">
+ <li>
+ <a href="#" onclick="openClose('import-form'); return false;"><i class="fa fa-cloud-upload generic-icons"></i> {{$import_label}}</a>
+ </li>
+ <form id="import-form" enctype="multipart/form-data" method="post" action="" style="display: none;" class="sub-menu">
+ <p class="descriptive-text">{{$file_import_text}}</p>
+ <div class="form-group">
+ <select id="import" name="target" class="form-control">
+ <option value="">{{$import_placeholder}}</option>
+ <!-- Loop through folders here -->
+ </select>
+ </div>
+ <!-- Or upload a zipped file containing the website -->
+ <p class="descriptive-text">{{$file_upload_text}}</p>
+ <div class="form-group">
+ <input id="website-upload-choose" type="file" name="zip_file" />
+ </div>
+ <button class="btn btn-primary btn-sm" type="submit" name="w_upload" value="w_upload">Upload</button>
+ </form>
+ </ul>
+</div>
diff --git a/view/tpl/wiki.tpl b/view/tpl/wiki.tpl
index 7617808f4..d9a4f8be3 100644
--- a/view/tpl/wiki.tpl
+++ b/view/tpl/wiki.tpl
@@ -311,6 +311,9 @@ function wiki_delete_wiki(wikiHtmlName, resource_id) {
ev.preventDefault();
return false;
}
+ if(!confirm('Are you sure you want to delete the page: ' + window.wiki_page_name)) {
+ return;
+ }
$.post("wiki/{{$channel}}/delete/page", {name: window.wiki_page_name, resource_id: window.wiki_resource_id},
function (data) {
if (data.success) {