372 lines
14 KiB
PHP
372 lines
14 KiB
PHP
<?php require $_SERVER['DOCUMENT_ROOT'].'/api/private/core.php';
|
|
|
|
use pizzaboxer\ProjectPolygon\Database;
|
|
use pizzaboxer\ProjectPolygon\Polygon;
|
|
use pizzaboxer\ProjectPolygon\Users;
|
|
use pizzaboxer\ProjectPolygon\Thumbnails;
|
|
use pizzaboxer\ProjectPolygon\Catalog;
|
|
use pizzaboxer\ProjectPolygon\Gzip;
|
|
use pizzaboxer\ProjectPolygon\PageBuilder;
|
|
|
|
Users::RequireLogin();
|
|
|
|
$PlaceCount = Database::singleton()->run("SELECT COUNT(*) FROM assets WHERE creator = :UserID AND type = 9", [":UserID" => SESSION["user"]["id"]])->fetchColumn();
|
|
if ($PlaceCount >= SESSION["user"]["PlaceSlots"])
|
|
{
|
|
PageBuilder::instance()->errorCode(200, [
|
|
"title" => "Maximum place slots reached",
|
|
"text" => "You have reached the maximum number of place slots. Update any spare existing place slots you may have."
|
|
]);
|
|
}
|
|
|
|
$TemplatePlaces = Database::singleton()->run("SELECT * FROM assets WHERE TemplateOrder IS NOT NULL ORDER BY TemplateOrder")->fetchAll();
|
|
|
|
$PlayerCounts = range(1, 20);
|
|
$Versions = [2010, 2011, 2012];
|
|
$Error = false;
|
|
|
|
function IsTemplatePlace($PlaceID)
|
|
{
|
|
global $TemplatePlaces;
|
|
|
|
foreach ($TemplatePlaces as $TemplatePlace)
|
|
{
|
|
if ($TemplatePlace["id"] == $PlaceID) return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
if ($_SERVER["REQUEST_METHOD"] == "POST")
|
|
{
|
|
$Name = $_POST["Name"] ?? "";
|
|
$Description = $_POST["Description"] ?? "";
|
|
$PlayerCount = $_POST["PlayerCount"] ?? 10;
|
|
$Access = $_POST["Access"] ?? "Everyone";
|
|
$Version = $_POST["Version"] ?? 2010;
|
|
$PlaceTemplate = $_POST["PlaceTemplateSelection"] ?? "none";
|
|
$ChatType = $_POST["ChatType"] ?? "Classic";
|
|
|
|
$Copylocked = ($_POST["Copylocked"] ?? "") == "on";
|
|
$CommentsAllowed = ($_POST["CommentsAllowed"] ?? "") == "on";
|
|
|
|
$PlaceUpload = $_FILES["PlaceUpload"] ?? false;
|
|
|
|
Catalog::ParseGearAttributes();
|
|
|
|
if (!strlen($Name))
|
|
{
|
|
$Error = "Place Name is required";
|
|
}
|
|
else if (strlen($Name) > 50)
|
|
{
|
|
$Error = "Place Name cannot be longer than 50 characters";
|
|
}
|
|
else if (Polygon::IsExplicitlyFiltered($Name))
|
|
{
|
|
$Error = "Place Name contains inappropriate text";
|
|
}
|
|
else if (strlen($Description) > 1000)
|
|
{
|
|
$Error = "Place Description cannot be longer than 1000 characters";
|
|
}
|
|
else if (Polygon::IsExplicitlyFiltered($Description))
|
|
{
|
|
$Error = "Place Description contains inappropriate text";
|
|
}
|
|
else if ($PlaceTemplate == "none")
|
|
{
|
|
$Error = "No Place Template has been selected";
|
|
}
|
|
else if ($PlaceTemplate != "custom" && !IsTemplatePlace($PlaceTemplate))
|
|
{
|
|
$Error = "Invalid Place Template selected";
|
|
}
|
|
else if (!in_array((int)$PlayerCount, $PlayerCounts))
|
|
{
|
|
$Error = "Maximum Visitor Count must be within 1 - 20 Players";
|
|
}
|
|
else if (!in_array((int)$Version, $Versions))
|
|
{
|
|
$Error = "Invalid Place Version selected";
|
|
}
|
|
else if (!in_array($Access, ["Everyone", "Friends"]))
|
|
{
|
|
$Error = "Invalid access level selected";
|
|
}
|
|
else if ($Access != "Friends" && $Version != "2011")
|
|
{
|
|
$Error = "2010 and 2012 places must be set to friends-only";
|
|
}
|
|
else if (!in_array($ChatType, ["Classic", "Bubble", "Both"]))
|
|
{
|
|
$Error = "Invalid Chat Type selected";
|
|
}
|
|
else if ($PlaceTemplate == "custom" && ($PlaceUpload === false || $PlaceUpload["size"] == 0))
|
|
{
|
|
$Error = "No Place File has been selected for upload";
|
|
}
|
|
else if ($PlaceTemplate == "custom" && $PlaceUpload["size"] > 32000000)
|
|
{
|
|
$Error = "Place File cannot be larger than 32 megabytes";
|
|
}
|
|
else
|
|
{
|
|
if ($PlaceTemplate == "custom")
|
|
{
|
|
$PlaceXML = file_get_contents($PlaceUpload["tmp_name"]);
|
|
$PlaceXML = str_ireplace("http://".$_SERVER['HTTP_HOST']."/asset/?id=", "%ASSETURL%", $PlaceXML);
|
|
$PlaceXML = str_ireplace("http://".$_SERVER['HTTP_HOST']."/asset?id=", "%ASSETURL%", $PlaceXML);
|
|
$PlaceXML = preg_replace("/rbxasset:\/\/..\/[^<]*/", "", $PlaceXML);
|
|
|
|
|
|
libxml_use_internal_errors(true);
|
|
$SimpleXML = simplexml_load_string($PlaceXML);
|
|
|
|
if ($SimpleXML === false)
|
|
{
|
|
// temporary hack
|
|
foreach (libxml_get_errors() as $XMLError)
|
|
{
|
|
// ignore "invalid xmlChar value" error
|
|
// this can trigger false positives as some scripts may use binary xml characters
|
|
if ($XMLError->code != 9)
|
|
{
|
|
$Error = "Place File is invalid, are you sure it is an older format place file?";
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!$Error)
|
|
{
|
|
$PlaceID = Catalog::CreateAsset([
|
|
"type" => 9,
|
|
"creator" => SESSION["user"]["id"],
|
|
"name" => $Name,
|
|
"description" => $Description,
|
|
"comments" => (int)$CommentsAllowed,
|
|
"PublicDomain" => (int)!$Copylocked,
|
|
"ServerRunning" => 0,
|
|
"MaxPlayers" => $PlayerCount,
|
|
"Access" => $Access,
|
|
"ActivePlayers" => 0,
|
|
"Visits" => 0,
|
|
"Version" => $Version,
|
|
"ChatType" => $ChatType,
|
|
"gear_attributes" => json_encode(Catalog::$GearAttributes),
|
|
"approved" => 1
|
|
]);
|
|
|
|
$PlaceLocation = Polygon::GetSharedResource("assets/{$PlaceID}");
|
|
|
|
if ($PlaceTemplate == "custom")
|
|
{
|
|
file_put_contents($PlaceLocation, $PlaceXML);
|
|
Gzip::Compress($PlaceLocation);
|
|
}
|
|
else
|
|
{
|
|
copy(Polygon::GetSharedResource("assets/{$PlaceTemplate}"), $PlaceLocation);
|
|
}
|
|
|
|
Polygon::RequestRender("Place", $PlaceID);
|
|
|
|
redirect("/" . encode_asset_name($Name) . "-place?id={$PlaceID}");
|
|
}
|
|
}
|
|
|
|
$pageBuilder = new PageBuilder(["title" => "Create Place"]);
|
|
$pageBuilder->buildHeader();
|
|
?>
|
|
<form method="post" enctype="multipart/form-data">
|
|
<h2 class="font-weight-normal">Create Place</h2>
|
|
<?php if($Error) { ?><div class="alert alert-danger px-2 py-1" role="alert"><?=$Error?></div><?php } ?>
|
|
<ul class="nav nav-tabs px-2" id="placesTabs" role="tablist">
|
|
<li class="nav-item">
|
|
<a class="nav-link active" id="templates-tab" data-toggle="tab" href="#templates" role="tab" aria-controls="templates" aria-selected="true">Templates</a>
|
|
</li>
|
|
<li class="nav-item">
|
|
<a class="nav-link" id="basic-settings-tab" data-toggle="tab" href="#basic-settings" role="tab" aria-controls="basic-settings" aria-selected="false">Basic Settings</a>
|
|
</li>
|
|
<li class="nav-item">
|
|
<a class="nav-link" id="advanced-settings-tab" data-toggle="tab" href="#advanced-settings" role="tab" aria-controls="advanced-settings" aria-selected="false">Advanced Settings</a>
|
|
</li>
|
|
</ul>
|
|
<div class="tab-content pt-2 mb-4" id="placesTabsContent">
|
|
<div class="tab-pane active" id="templates" role="tabpanel">
|
|
<h3 class="font-weight-normal mt-2 mb-4">Place Templates</h3>
|
|
<div class="row px-2">
|
|
<?php foreach ($TemplatePlaces as $TemplatePlace) { ?>
|
|
<div class="col-lg-3 col-md-4 col-6 px-2 mb-3">
|
|
<div class="place-template card hover h-100" role="button" data-template-id="<?=$TemplatePlace["id"]?>">
|
|
<img class="card-img-top img-fluid" title="<?=Polygon::FilterText($TemplatePlace["name"])?>" alt="<?=Polygon::FilterText($TemplatePlace["name"])?>" src="<?=Thumbnails::GetAsset((object)$TemplatePlace, 768, 432)?>">
|
|
<div class="card-body p-2 text-center">
|
|
<p class="mb-0 text-truncate" title="<?=Polygon::FilterText($TemplatePlace["name"])?>"><?=Polygon::FilterText($TemplatePlace["name"])?></p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<?php } ?>
|
|
<div class="col-lg-3 col-md-4 col-6 px-2 mb-3">
|
|
<div class="place-template card hover h-100" role="button" data-template-id="custom">
|
|
<img class="card-img-top img-fluid" title="Create from Place File" alt="Create from Place File" src="<?=Thumbnails::GetStatus("rendering", 768, 432)?>">
|
|
<div class="card-body p-2 text-center">
|
|
<p class="mb-0 text-truncate" title="Create from Place File">Create from Place File</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="tab-pane" id="basic-settings" role="tabpanel">
|
|
<h3 class="font-weight-normal mt-2 mb-4">Basic Settings</h3>
|
|
<div class="form-group" style="max-width: 26em;">
|
|
<label for="Name">Name:</label>
|
|
<input name="Name" id="Name" type="text" value="<?=SESSION["user"]["username"]?>'s Place<?=$PlaceCount >= 1 ? " Number: " . ($PlaceCount+1) : ""?>" maxlength="50" class="form-control form-control-sm">
|
|
</div>
|
|
<div class="form-group" style="max-width: 26rem;">
|
|
<label for="Description">Description:</label>
|
|
<textarea name="Description" id="Name" maxlength="1000" class="form-control form-control-sm" rows="5"></textarea>
|
|
</div>
|
|
<div class="form-group" style="max-width: 26rem;">
|
|
<label for="PlayerCount">Number of Players:</label>
|
|
<select name="PlayerCount" id="PlayerCount" class="form-control form-control-sm">
|
|
<?php foreach ($PlayerCounts as $PlayerCount) { ?>
|
|
<option value="<?=$PlayerCount?>"<?=$PlayerCount == 10 ? "selected=\"selected\"" : ""?>><?=$PlayerCount?></option>
|
|
<?php } ?>
|
|
</select>
|
|
</div>
|
|
<div class="form-group" style="max-width: 26rem;">
|
|
<label for="Access">Access: <i class="far fa-question-circle text-muted" title="2010 and 2012 places can only be accessible to friends for the time being" data-toggle="tooltip"></i></label>
|
|
<select name="Access" id="Access" class="form-control form-control-sm">
|
|
<option value="Everyone" disabled="disabled">Everyone</option>
|
|
<option value="Friends" selected="selected">Friends</option>
|
|
</select>
|
|
</div>
|
|
<div class="form-group" style="max-width: 26rem;">
|
|
<label for="Version">Version:</label>
|
|
<select name="Version" class="form-control form-control-sm" id="Version">
|
|
<?php foreach ($Versions as $Version) { ?>
|
|
<option value="<?=$Version?>"><?=$Version?></option>
|
|
<?php } ?>
|
|
</select>
|
|
</div>
|
|
</div>
|
|
<div class="tab-pane" id="advanced-settings" role="tabpanel">
|
|
<h3 class="font-weight-normal mt-2 mb-4">Gear Permissions</h3>
|
|
<label for="PlayerCount">Gear types:</label>
|
|
<div style="max-width: 32rem;" class="row ml-4">
|
|
<?php foreach (Catalog::$GearAttributes as $Gear => $GearEnabled) { ?>
|
|
<div class="col-sm-4 mb-1">
|
|
<div class="form-check">
|
|
<input type="checkbox" class="form-check-input" id="gear_<?=$Gear?>" name="gear_<?=$Gear?>">
|
|
<label class="form-check-label" for="gear_<?=$Gear?>"><?=Catalog::$GearAttributesDisplay[$Gear]["text_sel"]?></label>
|
|
</div>
|
|
</div>
|
|
<?php } ?>
|
|
</div>
|
|
<div class="divider-top my-4"></div>
|
|
<h3 class="font-weight-normal mt-2 mb-4">Other Permissions</h3>
|
|
<div class="form-group" style="max-width: 26rem;">
|
|
<label for="ChatType">Chat Type:</label>
|
|
<select name="ChatType" class="form-control form-control-sm" id="ChatType">
|
|
<option>Classic</option>
|
|
<option>Bubble</option>
|
|
<option>Both</option>
|
|
</select>
|
|
</div>
|
|
<div class="form-check">
|
|
<input type="checkbox" class="form-check-input" id="Copylocked" name="Copylocked" checked="checked">
|
|
<label class="form-check-label" for="Copylocked">Copy Locked</label>
|
|
</div>
|
|
<div class="form-check">
|
|
<input type="checkbox" class="form-check-input" id="CommentsAllowed" name="CommentsAllowed" checked="checked">
|
|
<label class="form-check-label" for="CommentsAllowed">Comments Allowed</label>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<button class="btn btn-success create-place" type="submit"><h5 class="font-weight-normal mb-0">Create Place</h5></button>
|
|
<button class="btn btn-secondary" onclick="window.history.back()"><h5 class="font-weight-normal mb-0">Cancel</h5></button>
|
|
<input name="PlaceTemplateSelection" type="hidden" value="none">
|
|
<input name="PlaceUpload" type="file" style="display:none">
|
|
</form>
|
|
|
|
<script>
|
|
$(".place-template").click(function()
|
|
{
|
|
var PlaceTemplateID = $(this).attr("data-template-id");
|
|
|
|
if (PlaceTemplateID == "custom")
|
|
{
|
|
polygon.buildModal({
|
|
header: "Upload Place File",
|
|
body: "<p class=\"text-left\">Here, you can upload a Place file manually without having to use Studio. If you are unable to upload with Studio, you can use this instead.</p>",
|
|
buttons: [{class:"btn btn-primary px-4", dismiss:true, text:"OK"}],
|
|
options: {show: true, backdrop: "static"}
|
|
});
|
|
|
|
$(".global.modal .modal-body").append($("input[name='PlaceUpload']"));
|
|
$("input[name='PlaceUpload']").show();
|
|
}
|
|
else
|
|
{
|
|
$("input[name='PlaceUpload']").val("");
|
|
|
|
$("input[name='PlaceTemplateSelection']").val(PlaceTemplateID);
|
|
$(".place-template[data-template-id='custom']").find(".card-body p").text("Create from Place File");
|
|
|
|
$(".place-template").children().removeClass("bg-primary");
|
|
$(this).find(".card-body").addClass("bg-primary");
|
|
}
|
|
});
|
|
|
|
$("input[name='PlaceUpload']").change(function(event)
|
|
{
|
|
var FileName = event.target.files[0].name;
|
|
|
|
$("input[name='PlaceTemplateSelection']").val("custom");
|
|
$(".place-template[data-template-id='custom']").find(".card-body p").text(FileName);
|
|
|
|
$(".place-template").children().removeClass("bg-primary");
|
|
$(".place-template[data-template-id='custom']").find(".card-body").addClass("bg-primary");
|
|
});
|
|
|
|
$("select[name='Version']").change(function()
|
|
{
|
|
if ($(this).val() == "2010" || $(this).val() == "2012")
|
|
{
|
|
$("select[name='Access']").val("Friends");
|
|
$("select[name='Access'] option[value='Everyone']").attr("disabled", "disabled");
|
|
$("label[for='Access'] i").removeClass("d-none");
|
|
}
|
|
else
|
|
{
|
|
$("label[for='Access'] i").addClass("d-none");
|
|
$("select[name='Access'] option[value='Everyone']").removeAttr("disabled");
|
|
}
|
|
});
|
|
|
|
$('.global.modal').on('hidden.bs.modal', function()
|
|
{
|
|
$("input[name='PlaceUpload']").hide();
|
|
$("form").append($("input[name='PlaceUpload']"));
|
|
});
|
|
|
|
$(".create-place").click(function(event)
|
|
{
|
|
event.preventDefault();
|
|
$(this).attr("disabled", "disabled");
|
|
|
|
$("input[name='PlaceUpload']").hide();
|
|
$("form").append($("input[name='PlaceUpload']"));
|
|
|
|
polygon.buildModal({options: {show: true, backdrop: "static"}});
|
|
$(".modal-content").hide();
|
|
$(".modal-dialog").append('<div class="processing text-center m-auto text-white"><span class="spinner-border" style="width: 4rem; height: 4rem; display: inline-block;" role="status"></span> <h4 class="font-weight-normal"> creating place, please wait...</h4></div>');
|
|
|
|
$("form").submit();
|
|
});
|
|
</script>
|
|
<?php $pageBuilder->buildFooter(); ?>
|