Penyebab Masalah:
- Croppie.js mengembalikan format base64 yang tidak sesuai dengan yang diharapkan sistem
- Validasi mime type gagal karena format base64 yang dikirim tidak dikenali
- Ekstensi file tidak cocok dengan allowed images
Solusi kode lengkap: admin/modules/system/app_user.php
<?php
/**
* Copyright (C) 2007,2008 Arie Nugraha (dicarve@yahoo.com)
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*
*/
use SLiMS\Filesystems\Storage;
/* Staffs/Application Users Management section */
// key to authenticate
define('INDEX_AUTH', '1');
// main system configuration
require '../../../sysconfig.inc.php';
// IP based access limitation
require LIB.'ip_based_access.inc.php';
do_checkIP('smc');
do_checkIP('smc-system');
// start the session
require SB.'admin/default/session.inc.php';
require SB.'admin/default/session_check.inc.php';
require SIMBIO.'simbio_GUI/form_maker/simbio_form_table_AJAX.inc.php';
require SIMBIO.'simbio_GUI/table/simbio_table.inc.php';
require SIMBIO.'simbio_GUI/paging/simbio_paging.inc.php';
require SIMBIO.'simbio_DB/datagrid/simbio_dbgrid.inc.php';
require SIMBIO.'simbio_DB/simbio_dbop.inc.php';
require SIMBIO.'simbio_FILE/simbio_file_upload.inc.php';
// privileges checking
$can_read = utility::havePrivilege('system', 'r');
$can_write = utility::havePrivilege('system', 'w');
function getUserType($obj_db, $array_data, $col) {
global $sysconf;
if (isset($sysconf['system_user_type'][$array_data[$col]])) {
return $sysconf['system_user_type'][$array_data[$col]];
}
}
// check if we want to change current user profile
$changecurrent = false;
if (isset($_GET['changecurrent'])) {
$changecurrent = true;
}
if (!$changecurrent) {
// only administrator have privileges add/edit users
if ($_SESSION['uid'] != 1) {
die('<div class="errorBox">'.__('You don\'t have enough privileges to view this section').'</div>');
}
}
/**
* Verify Two-factor authentication (2FA)
*/
if (isset($_POST['secret_code']) && isset($_POST['verify_code'])) {
$secret = utility::filterData('secret_code', 'post', true, true, true);
$otp = OTPHP\TOTP::createFromSecret($secret);
$verify_code = trim(utility::filterData('verify_code', 'post', true, true, true));
$isOk = $otp->verify($verify_code);
if ($isOk) {
// save to session for next purpose
$_SESSION['2fa_secret'] = $secret;
toastr(__('Code verified!'))->success();
} else {
unset($_SESSION['2fa_secret']);
toastr(__('Invalid verification code!'))->error();
}
exit;
}
if (isset($_POST['updateRecordID']) && isset($_POST['disable2fa']) && isset($_GET['diable2fa'])) {
$uid = (int)utility::filterData('updateRecordID', 'post', true, true, true);
$arr = explode(':', $uid);
if ($_SESSION['uid'] == 1 || $uid == $_SESSION['uid']) {
$update = $dbs->query(sprintf("update user set 2fa = null where user_id = '%d'", $uid));
if ($update) {
echo '<script type="text/javascript">parent.$(\'#mainContent\').simbioAJAX(parent.$.ajaxHistory[0].url);</script>';
toastr(__('Two-factor authentication has been disabled.'))->success();
} else {
toastr(__('Upss... something wrong!'))->error();
}
} else {
toastr(__('You don\'t have enough privileges to view this section'))->error();
}
exit;
}
/* REMOVE IMAGE */
if (isset($_POST['removeImage']) && isset($_POST['uimg']) && isset($_POST['img'])) {
// validate post image
$user_id = $_SESSION['uid'] > 1 ? (integer)$_SESSION['uid'] : (integer)utility::filterData('uimg', 'post', true, true, true);
$image_name = $dbs->escape_string(utility::filterData('img', 'post', true, true, true));
$query_image = $dbs->query("SELECT user_id FROM user WHERE user_id='{$user_id}' AND user_image='{$image_name}'");
if ($query_image->num_rows > 0) {
$_delete = $dbs->query(sprintf('UPDATE user SET user_image=NULL WHERE user_id=%d', $user_id));
if ($_delete) {
// Change upict
$_SESSION['upict'] = 'person.png';
$postImage = stripslashes($_POST['img']);
$postImage = str_replace('/', '', $postImage);
$imageDisk = Storage::images();
$imagePath = sprintf('persons/%s', $postImage);
if (!empty($postImage) && $imageDisk->isExists($imagePath)) {
@$imageDisk->delete($imagePath);
}
exit('<script type="text/javascript">alert(\''.str_replace('{imageFilename}', $postImage, __('{imageFilename} successfully removed!')).'\'); $(\'#userImage, #imageFilename\').remove();</script>');
}
}
exit();
}
/* RECORD OPERATION */
if (isset($_POST['saveData'])) { //echo '<pre>'; var_dump($_SESSION); echo '</pre>'; die();
$userName = $_SESSION['uid'] > 1 ? $_SESSION['uname'] : trim(strip_tags($_POST['userName']));
$realName = trim(strip_tags($_POST['realName']));
$passwd1 = $dbs->escape_string(trim($_POST['passwd1']));
$passwd2 = $dbs->escape_string(trim($_POST['passwd2']));
$old_user_image = $_POST['old_user_image'] ?? '';
// check form validity
if (empty($userName) OR empty($realName)) {
toastr(__('User Name or Real Name can\'t be empty'))->error();
exit();
} else if (($userName == 'admin' OR $realName == 'Administrator') AND $_SESSION['uid'] != 1) {
toastr(__('Login username or Real Name is probihited!'))->error();
exit();
} else if ($sysconf['password_policy_strong'] && ($passwd1 AND $passwd2) && ($passwd1 === $passwd2) && !simbio_security::validatePassword($passwd2, $sysconf['password_policy_min_length'])) {
toastr(__( sprintf('Password should at least %d characters long, contains one capital letter, one number, and one non-alphanumeric character !', $sysconf['password_policy_min_length']) ))->error();
exit();
} else if (($passwd1 AND $passwd2) AND ($passwd1 !== $passwd2)) {
toastr(__('Password confirmation does not match. See if your Caps Lock key is on!'))->error();
exit();
} else if (!simbio_form_maker::isTokenValid()) {
toastr(__('Invalid form submission token!'))->error();
exit();
} else {
$data['username'] = $dbs->escape_string(trim($userName));
$data['realname'] = $dbs->escape_string(trim($realName));
$data['user_type'] = (integer)$_POST['userType'];
$data['email'] = $dbs->escape_string(trim($_POST['eMail']));
$social_media = array();
foreach ($_POST['social'] as $id => $social) {
$social_val = $dbs->escape_string(trim($social));
if ($social_val != '') {
$social_media[$id] = $social_val;
}
}
if ($social_media) {
$data['social_media'] = $dbs->escape_string(serialize($social_media));
}
// only update group data if the flag is set, and user have enough privileges
if (isset($_POST['noChangeGroup']) AND !$changecurrent AND $can_read AND $can_write) {
// parsing groups data
$groups = '';
if (isset($_POST['groups']) AND !empty($_POST['groups'])) {
$groups = serialize($_POST['groups']);
} else {
$groups = 'literal{NULL}';
}
$data['groups'] = trim($groups);
}
if (($passwd1 AND $passwd2) AND ($passwd1 === $passwd2)) {
if ( (isset($_GET['changecurrent'])) AND ($_GET['changecurrent']='true') ) {
$old_passwd = $dbs->escape_string(trim($_POST['old_passwd']));
$up_q = $dbs->query('SELECT passwd FROM user WHERE user_id='.$_SESSION['uid']);
$up_d = $up_q->fetch_row();
if (password_verify($old_passwd, $up_d[0])) {
$data['passwd'] = password_hash($passwd2, PASSWORD_BCRYPT);
} else {
toastr(__('Password change failed. Make sure you input the old password.'))->error();
exit();
}
} else {
$data['passwd'] = password_hash($passwd2, PASSWORD_BCRYPT);
}
}
$data['input_date'] = date('Y-m-d');
$data['last_update'] = date('Y-m-d');
// save 2fa secret to database
if (isset($_SESSION['2fa_secret'])) {
$data['2fa'] = $_SESSION['2fa_secret'];
unset($_SESSION['2fa_secret']);
}
$imageDisk = Storage::images();
$new_filename_base = 'user_'.str_replace(array(',', '.', ' ', '-'), '_', strtolower($data['username']));
$base64_data = null;
$file_extension = 'jpg';
$upload_success = false;
if (!empty($_POST['base64picstring'])) {
$base64_full_string = $_POST['base64picstring'];
$file_extension = 'jpg'; // default
$base64_data_raw = '';
if (strpos($base64_full_string, '#image/type#') !== false) {
$parts = explode('#image/type#', $base64_full_string);
$base64_data_raw = $parts[0];
$file_extension = isset($parts[1]) ? strtolower(trim($parts[1])) : 'jpg';
// Clean base64 string - remove any non-base64 characters
$base64_data_raw = preg_replace('/[^A-Za-z0-9+\/=]/', '', $base64_data_raw);
// Validasi ekstensi file
$allowed_exts = array_map(function($ext) { return ltrim($ext, '.'); }, $sysconf['allowed_images']);
if (!in_array($file_extension, $allowed_exts)) {
utility::jsToastr('System User', __('Tipe file tidak diizinkan. Gunakan: ') . implode(', ', $sysconf['allowed_images']), 'error');
} else {
// Decode base64
$filedata = base64_decode($base64_data_raw, true);
if ($filedata !== false && $filedata !== null && strlen($filedata) > 0) {
// Get mime type from file data
$finfo = finfo_open(FILEINFO_MIME_TYPE);
$mime_type = finfo_buffer($finfo, $filedata);
finfo_close($finfo);
// Validasi mime type
$allowed_mimes = $sysconf['allowed_images_mimetype'];
$is_valid_mime = in_array($mime_type, $allowed_mimes);
// Validasi ukuran file
$file_size = strlen($filedata);
$max_size = $sysconf['max_image_upload'] * 1024;
$is_valid_size = $file_size <= $max_size;
if ($is_valid_mime && $is_valid_size) {
$new_filename = $new_filename_base . '.' . $file_extension;
// METODE LANGSUNG (TANPA STORAGE CLASS)
$uploadDir = SB . 'images' . DS . 'persons' . DS;
if (!is_dir($uploadDir)) {
mkdir($uploadDir, 0777, true);
}
$fullFilePath = $uploadDir . $new_filename;
if (file_put_contents($fullFilePath, $filedata)) {
if (!empty($old_user_image) && $old_user_image != $new_filename) {
$oldFilePath = $uploadDir . $old_user_image;
if (file_exists($oldFilePath)) {
@unlink($oldFilePath);
}
}
$data['user_image'] = $dbs->escape_string($new_filename);
$upload_success = true;
} else {
utility::jsToastr('System User', __('Gagal menyimpan file gambar'), 'error');
}
} else {
$error_msg = [];
if (!$is_valid_mime) $error_msg[] = __('Tipe file tidak valid: ') . $mime_type;
if (!$is_valid_size) $error_msg[] = __('Ukuran file melebihi ') . $sysconf['max_image_upload'] . ' KB';
utility::jsToastr('System User', __('Gambar Gagal Diupload') . '<br/>' . implode('<br/>', $error_msg), 'error');
}
} else {
utility::jsToastr('System User', __('Gambar Gagal Diupload') . '<br/>' . __('Data base64 tidak valid - decode failed'), 'error');
}
}
} elseif (strpos($base64_full_string, 'base64,') !== false) {
// Handle format base64 standar dari webcam
list($mime, $data_string) = explode(';', $base64_full_string);
list(, $base64_data_raw) = explode(',', $data_string);
$base64_data_raw = preg_replace('/[^A-Za-z0-9+\/=]/', '', $base64_data_raw);
$mime_type = str_replace('data:', '', $mime);
$file_extension = 'jpg';
if ($mime_type == 'image/png') $file_extension = 'png';
if ($mime_type == 'image/gif') $file_extension = 'gif';
if ($mime_type == 'image/jpeg') $file_extension = 'jpg';
$filedata = base64_decode($base64_data_raw, true);
if ($filedata !== false && $filedata !== null && strlen($filedata) > 0) {
$new_filename = $new_filename_base . '.' . $file_extension;
$uploadDir = SB . 'images' . DS . 'persons' . DS;
if (!is_dir($uploadDir)) {
mkdir($uploadDir, 0777, true);
}
$fullFilePath = $uploadDir . $new_filename;
if (file_put_contents($fullFilePath, $filedata)) {
if (!empty($old_user_image) && $old_user_image != $new_filename) {
$oldFilePath = $uploadDir . $old_user_image;
if (file_exists($oldFilePath)) {
@unlink($oldFilePath);
}
}
$data['user_image'] = $dbs->escape_string($new_filename);
$upload_success = true;
}
} else {
utility::jsToastr('System User', __('Gambar Gagal Diupload') . '<br/>' . __('Data base64 tidak valid - webcam capture'), 'error');
}
} else {
$base64_data_raw = preg_replace('/[^A-Za-z0-9+\/=]/', '', $base64_full_string);
$filedata = base64_decode($base64_data_raw, true);
if ($filedata !== false && $filedata !== null && strlen($filedata) > 0) {
$new_filename = $new_filename_base . '.jpg';
$uploadDir = SB . 'images' . DS . 'persons' . DS;
if (!is_dir($uploadDir)) {
mkdir($uploadDir, 0777, true);
}
$fullFilePath = $uploadDir . $new_filename;
if (file_put_contents($fullFilePath, $filedata)) {
if (!empty($old_user_image) && $old_user_image != $new_filename) {
$oldFilePath = $uploadDir . $old_user_image;
if (file_exists($oldFilePath)) {
@unlink($oldFilePath);
}
}
$data['user_image'] = $dbs->escape_string($new_filename);
$upload_success = true;
}
} else {
utility::jsToastr('System User', __('Gambar Gagal Diupload') . '<br/>' . __('Format base64 tidak dikenal'), 'error');
}
}
}
elseif (!empty($_FILES['image']) AND $_FILES['image']['size'] > 0) {
$uploadDir = SB . 'images' . DS . 'persons' . DS;
if (!is_dir($uploadDir)) {
mkdir($uploadDir, 0777, true);
}
$fileExt = pathinfo($_FILES['image']['name'], PATHINFO_EXTENSION);
$new_filename = $new_filename_base . '.' . strtolower($fileExt);
$fullFilePath = $uploadDir . $new_filename;
if (move_uploaded_file($_FILES['image']['tmp_name'], $fullFilePath)) {
if (!empty($old_user_image) && $old_user_image != $new_filename) {
$oldFilePath = $uploadDir . $old_user_image;
if (file_exists($oldFilePath)) {
@unlink($oldFilePath);
}
}
$data['user_image'] = $dbs->escape_string($new_filename);
$upload_success = true;
} else {
utility::jsToastr('System User', __('Image FAILED to upload'), 'error');
}
}
// Set upload_status untuk notifikasi - TANPA define()
if ($upload_success) {
$upload_status = 1; // Langsung assign nilai, tanpa mendefinisikan konstanta
}
// create sql op object
$sql_op = new simbio_dbop($dbs);
if (isset($_POST['updateRecordID'])) {
/* UPDATE RECORD MODE */
// remove input date
unset($data['input_date']);
// filter update record ID
$updateRecordID = (integer)$_POST['updateRecordID'];
if ($_SESSION['uid'] != 1 && $updateRecordID !== (integer)$_SESSION['uid']) {
toastr(__('You don\'t have enough privileges to modify this user.'))->error();
exit();
}
// update the data
$update = $sql_op->update('user', $data, 'user_id='.$updateRecordID);
if ($update) {
// write log
writeLog('staff', $_SESSION['uid'], 'system', $_SESSION['realname'].' update user data ('.$data['realname'].') with username ('.$data['username'].')', 'User', 'Update');
toastr(__('User Data Successfully Updated'))->success();
// upload status alert
if (isset($upload_status)) {
if ($upload_status == UPLOAD_SUCCESS) {
// Change upict
$_SESSION['upict'] = $data['user_image'];
// write log
writeLog('staff', $_SESSION['uid'], 'system/user', $_SESSION['realname'].' upload image file '.$data['user_image'], 'User image', 'Upload');
toastr(__('Image Uploaded Successfully'))->success();
} else {
// write log
$log_error_msg = isset($upload) ? $upload->getError() : 'Base64/Validation failed';
writeLog('staff', $_SESSION['uid'], 'system/user', 'ERROR : '.$_SESSION['realname'].' FAILED TO upload image file (Error: '.$log_error_msg.')', 'User image', 'Fail');
}
}
echo '<script type="text/javascript">parent.$(\'#mainContent\').simbioAJAX(\''.$_SERVER['PHP_SELF'].'?'.$_SERVER['QUERY_STRING'].'\');</script>';
} else {
toastr(__('User Data FAILED to Updated. Please Contact System Administrator'))->error();
}
exit();
} else {
/* INSERT RECORD MODE */
// insert the data
if ($sql_op->insert('user', $data)) {
// write log
writeLog('staff', $_SESSION['uid'], 'system', $_SESSION['realname'].' add new user ('.$data['realname'].') with username ('.$data['username'].')', 'User', 'Add');
toastr(__('New User Data Successfully Saved'))->success();
// upload status alert
if (isset($upload_status)) {
if ($upload_status == UPLOAD_SUCCESS) {
// Change upict
$_SESSION['upict'] = $data['user_image'];
// write log
writeLog('staff', $_SESSION['uid'], 'system/user', $_SESSION['realname'].' upload image file '.$data['user_image'], 'User image', 'Upload');
toastr(__('Image Uploaded Successfully'))->success();
} else {
// write log
$log_error_msg = isset($upload) ? $upload->getError() : 'Base64/Validation failed';
writeLog('staff', $_SESSION['uid'], 'system/user', 'ERROR : '.$_SESSION['realname'].' FAILED TO upload image file (Error: '.$log_error_msg.')', 'User image', 'Fail');
}
}
echo '<script type="text/javascript">parent.$(\'#mainContent\').simbioAJAX(\''.$_SERVER['PHP_SELF'].'\');</script>';
} else {
toastr(__('User Data FAILED to Save. Please Contact System Administrator'))->error();
}
exit();
}
}
exit();
} else if (isset($_POST['itemID']) AND !empty($_POST['itemID']) AND isset($_POST['itemAction'])) {
if (!($can_read AND $can_write)) {
die();
}
/* DATA DELETION PROCESS */
$sql_op = new simbio_dbop($dbs);
$failed_array = array();
$error_num = 0;
if (!is_array($_POST['itemID'])) {
// make an array
$_POST['itemID'] = array((integer)$_POST['itemID']);
}
// loop array
foreach ($_POST['itemID'] as $itemID) {
$itemID = (integer)$itemID;
// get user data
$user_q = $dbs->query('SELECT username, realname FROM user WHERE user_id='.$itemID);
$user_d = $user_q->fetch_row();
if (!$sql_op->delete('user', "user_id='$itemID'")) {
$error_num++;
} else {
// write log
writeLog('staff', $_SESSION['uid'], 'system', $_SESSION['realname'].' DELETE user ('.$user_d[1].') with username ('.$user_d[0].')', 'User', 'Delete');
}
}
// error alerting
if ($error_num == 0) {
toastr(__('All Data Successfully Deleted'))->success();
echo '<script type="text/javascript">parent.$(\'#mainContent\').simbioAJAX(\''.$_SERVER['PHP_SELF'].'?'.$_POST['lastQueryStr'].'\');</script>';
} else {
toastr(__('Some or All Data NOT deleted successfully!\nPlease contact system administrator'))->error();
echo '<script type="text/javascript">parent.$(\'#mainContent\').simbioAJAX(\''.$_SERVER['PHP_SELF'].'?'.$_POST['lastQueryStr'].'\');</script>';
}
exit();
}
/* RECORD OPERATION END */
if (!$changecurrent) {
/* search form */
?>
<div class="menuBox">
<div class="menuBoxInner userIcon">
<div class="per_title">
<h2><?php echo __('Librarian & System Users'); ?></h2>
</div>
<div class="sub_section">
<div class="btn-group">
<a href="<?php echo MWB; ?>/system/app_user.php" class="btn btn-default"><?php echo __('User List'); ?></a>
<a href="<?php echo MWB; ?>system/app_user.php?action=detail" class="btn btn-default"><?php echo __('Add New User'); ?></a>
</div>
<form name="search" action="<?php echo MWB; ?>system/app_user.php" id="search" method="get" class="form-inline"><?php echo __('Search'); ?>
<input type="text" name="keywords" class="form-control col-md-3" />
<input type="submit" id="doSearch" value="<?php echo __('Search'); ?>" class="btn btn-default" />
</form>
</div>
</div>
</div>
<?php
/* search form end */
}
/* main content */
if (isset($_POST['detail']) OR (isset($_GET['action']) AND $_GET['action'] == 'detail')) {
if (!($can_read AND $can_write) AND !$changecurrent) {
die('<div class="errorBox">'.__('You don\'t have enough privileges to view this section').'</div>');
}
/* RECORD FORM */
// try query
$itemID = (integer)isset($_POST['itemID'])?$_POST['itemID']:0;
if ($changecurrent) {
$itemID = (integer)$_SESSION['uid'];
}
$rec_q = \SLiMS\DB::query('SELECT * FROM user WHERE user_id=?', [$itemID]);
$rec_d = $rec_q->first();
// create new instance
$form = new simbio_form_table_AJAX('mainForm', $_SERVER['PHP_SELF'].'?'.$_SERVER['QUERY_STRING'], 'post');
$form->submit_button_attr = 'name="saveData" value="'.__('Save').'" class="btn btn-default"';
// form table attributes
$form->table_attr = 'id="dataList" class="s-table table"';
$form->table_header_attr = 'class="alterCell font-weight-bold"';
$form->table_content_attr = 'class="alterCell2"';
// edit mode flag set
if ($rec_q->count() > 0) {
$form->edit_mode = true;
// record ID for delete process
if (!$changecurrent) {
// form record id
$form->record_id = $itemID;
} else {
$form->addHidden('updateRecordID', $itemID);
$form->back_button = false;
}
// form record title
$form->record_title = $rec_d['realname'];
// submit button attribute
$form->submit_button_attr = 'name="saveData" value="'.__('Update').'" class="btn btn-default"';
$form->addHidden('old_user_image', $rec_d['user_image'] ?? '');
}
/* Form Element(s) */
// user name
if ($_SESSION['uid'] > 1) {
$form->addAnything(__('Login Username'), '<strong>'.$rec_d['username'].'</strong>');
} else {
$form->addTextField('text', 'userName', __('Login Username').'*', $rec_d['username']??'', 'style="width: 50%;" class="form-control"');
}
// user real name
$form->addTextField('text', 'realName', __('Real Name').'*', $rec_d['realname']??'', 'style="width: 50%;" class="form-control"');
// user type
$utype_options = array();
foreach ($sysconf['system_user_type'] as $id => $name) {
$utype_options[] = array($id, $name);
}
$form->addSelectList('userType', __('User Type').'*', $utype_options, $rec_d['user_type']??'','class="form-control col-3"');
// user e-mail
$form->addTextField('text', 'eMail', __('E-Mail'), $rec_d['email']??'', 'style="width: 50%;" class="form-control"');
// social media link
$str_input = '';
$social_media = array();
if (isset($rec_d['social_media'])) {
$social_media = @unserialize($rec_d['social_media']);
}
$str_input = '<div class="row">';
foreach ($sysconf['social'] as $id => $social) {
$str_input .= '<div class="social-input col-4"><span class="social-form"><input type="text" name="social['.$id.']" value="'.(isset($social_media[$id])?$social_media[$id]:'').'" placeholder="'.$social.'" class="form-control" /></span></div>'."\n";
}
$str_input .= '</div>';
$form->addAnything(__('Social Media'), $str_input);
$imageDisk = Storage::images();
$str_input = '<div class="row">';
$str_input .= '<div class="col-2">';
$str_input .= '<div id="imageFilename" class="s-margin__bottom-1">';
if (isset($rec_d['user_image']) && $imageDisk->isExists('persons/'.$rec_d['user_image'])) { // Check existence using Storage
$str_input .= '<a href="'.SWB . 'lib/minigalnano/createthumb.php?filename=images/persons/' . urlencode($rec_d['user_image']) . '&width=600" class="openPopUp notAJAX" title="'.__('Click to enlarge preview').'" width="300" height="400" >';
$str_input .= '<img src="'.SWB.'lib/minigalnano/createthumb.php?filename=images/persons/'.urlencode(($rec_d['user_image']??'photo.png')).'&width=600&v='.date('this').'" class="img-fluid rounded" id="current_image_preview" alt="Image cover">';
$str_input .= '</a>';
// Tombol Remove
$str_input .= '<a href="'.MWB.'system/app_user.php" postdata="removeImage=true&uimg='.$itemID.'&img='.($rec_d['user_image']??'photo.png').'" loadcontainer="imageFilename" class="s-margin__bottom-1 s-btn btn btn-danger btn-block rounded-0 makeHidden removeImage">'.__('Remove Image').'</a>';
} else {
$str_input .= '<img src="'.SWB.'images/persons/person.png'.'?'.date('this').'" class="img-fluid rounded" id="current_image_preview" alt="Image cover">';
}
$str_input .= '</div>';
$str_input .= '</div>';
$str_input .= '<div class="custom-file col-4">';
$str_input .= simbio_form_element::textField('file', 'image', '', 'id="image" class="custom-file-input" accept="'.implode(',', $sysconf['allowed_images']).'"');
$str_input .= '<label class="custom-file-label" for="image">'.__('Choose file').'</label>';
$str_input .= '</div>';
$str_input .= ' <div class="mt-2 ml-2">'.__('Maximum').' '.$sysconf['max_image_upload'].' KB</div>';
$str_input .= '</div>';
if ($sysconf['webcam'] !== false) {
$str_input .= '<textarea id="base64picstring" name="base64picstring" style="display: none;"></textarea>';
if ($sysconf['webcam'] == 'flex') {
$str_input .= '<object id="flash_video" classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000" height="280px" width="100%">';
$str_input .= '<param name="src" value="'.SWB.'lib/flex/ShotSLiMSMemberPicture.swf"/>';
$str_input .= '<embed name="src" src="'.SWB.'lib/flex/ShotSLiMSMemberPicture.swf" height="280px" width="100%"/>';
$str_input .= '</object>';
} elseif ($sysconf['webcam'] == 'html5') {
$str_input .= '<div class="makeHidden_">';
$str_input .= '<p>'.__('or take a photo').'</p>';
$str_input .= '<div class="form-inline">';
$str_input .= '<div class="form-group pr-2">';
$str_input .= '<button id="btn_load" type="button" class="btn btn-primary" onclick="loadcam(this)">'.__('Load Camera').'</button>';
$str_input .= '</div>';
$str_input .= '<div class="form-group pr-2">';
$str_input .= '<select class="form-control" onchange="aspect(this)"><option value="1">1x1</option><option value="2" selected>2x3</option><option value="3">3x4</option></select>';
$str_input .= '</div>';
$str_input .= '<div class="form-group pr-2">';
$str_input .= '<select class="form-control" id="cmb_format" onchange="if(pause){set();}"><option value="png">PNG</option><option value="jpg">JPEG</option></select>';
$str_input .= '</div>';
$str_input .= '<div class="form-group pr-2">';
$str_input .= '<button id="btn_pause" type="button" class="btn btn-primary" onclick="snapshot(this)" disabled>'.__('Capture').'</button>';
$str_input .= '</div>';
$str_input .= '<div class="form-group pr-2">';
$str_input .= '<button type="button" id="btn_reset" class="btn btn-danger" onclick="resetvalue()">'.__('Reset').'</button>';
$str_input .= '</div>';
$str_input .= '</div>';
$str_input .= '<div id="my_container" class="makeHidden_ mt-2" style="width: 400px; height: 300px; border: 1px solid #f4f4f4; position: relative;">';
$str_input .= '<video id="my_vid" autoplay width="400" height="300" style="float: left; position: absolute; left: 10;"></video>';
$str_input .= '<canvas id="my_canvas" width="400" height="300" style="float: left; position: absolute; left: 10; visibility: hidden;"></canvas>';
$str_input .= '<div id="my_frame" style="border: 1px solid #CCC; width: 160px; height: 240px; z-index: 2; margin: auto; position: absolute; top: 0; bottom: 0; left: 0; right: 0;"></div></div>';
$str_input .= '<canvas id="my_preview" width="160" height="240" style="width: 160px; height: 240px; border: 1px solid #f4f4f4; display: none;"></canvas>';
}
}
$form->addAnything(__('User Photo'), $str_input);
// user group
// only appear by user who hold system module privileges
if (!$changecurrent AND $can_read AND $can_write) {
// add hidden element as a flag that we dont change group data
$form->addHidden('noChangeGroup', '1');
// user group
$group_query = $dbs->query('SELECT group_id, group_name FROM
user_group WHERE group_id != 1');
// initiliaze group options
$group_options = array();
while ($group_data = $group_query->fetch_row()) {
$group_options[] = array($group_data[0], $group_data[1]);
}
$form->addCheckBox('groups', __('Group(s)'), $group_options, unserialize($rec_d['groups']??''));
}
// user password
if ( (isset($_GET['changecurrent'])) AND ($_GET['changecurrent']='true') ) {
$form->addTextField('password', 'old_passwd', __('Old Password').'*', '', 'style="width: 50%;" class="form-control"');
}
$form->addTextField('password', 'passwd1', __('New Password').'*', '', 'style="width: 50%;" class="form-control"');
// user password confirm
$form->addTextField('password', 'passwd2', __('Confirm New Password').'*', '', 'style="width: 50%;" class="form-control"');
// Two Factor Authentication
if (!empty($rec_d) && $rec_d['user_id'] === $_SESSION['uid'] && extension_loaded('iconv')) {
$otp = OTPHP\TOTP::generate();
$otp->setLabel(config('library_name'));
$secret = $otp->getSecret();
// generate qrcode
$render = new BaconQrCode\Renderer\ImageRenderer(
new BaconQrCode\Renderer\RendererStyle\RendererStyle(150),
new BaconQrCode\Renderer\Image\SvgImageBackEnd()
);
$writer = new BaconQrCode\Writer($render);
$qrcode = $writer->writeString($otp->getProvisioningUri());
$otp_html = '';
if (($rec_d['2fa'] ?? false)) {
$otp_html .= '<div class="alert alert-success d-flex justify-content-between"><span>🔐 ' . __('Two Factor Authentication enabled.') . '</span><div><button formaction="'.$_SERVER['PHP_SELF'].'?diable2fa=1&changecurrent=1" type="submit" name="disable2fa" value="1" class="btn btn-danger btn-sm">' . __('Disable It') . '</button></div></div>';
}
list($otp_app, $verification_code, $verify) = [
str_replace('{link}', '<a href="https://play.google.com/store/apps/details?id=org.fedorahosted.freeotp" target="_blank">FreeOTP</a>', __('Scan this QRcode with your authenticator (e.g. Google Authenticator or {link}) <br> and enter verification code below to enable Two Factor Authentication.')),
__('Verification code'),
__('Verify')
];
$otp_html .= <<<HTML
<div style="display:flex; align-items:center">
<div>{$qrcode}</div>
<div>
<div class="my-3">
{$otp_app}
</div>
<input form="formVerify2fa" type="hidden" name="secret_code" value="{$secret}">
<div class="text-muted">{$verification_code}</div>
<div class="input-group mb-3">
<input form="formVerify2fa" type="text" name="verify_code" class="form-control mr-0" placeholder="Enter code from authenticator" aria-label="Enter code from authenticator" aria-describedby="button-addon2">
<div class="input-group-append">
<button form="formVerify2fa" class="btn btn-outline-secondary" type="submit" id="button-addon2">{$verify}</button>
</div>
</div>
</div>
</div>
HTML;
$form->addAnything(__('Enable Two Factor Authentication'), $otp_html);
}
// ability to disable Two Factor Authentication for administrator
if ($_SESSION['uid'] == 1 && ($rec_d['2fa'] ?? false)) {
$otp_html = '<button formaction="'.$_SERVER['PHP_SELF'].'?diable2fa=1" type="submit" name="disable2fa" value="1" class="btn btn-danger">'.__('Disable Two Factor Authentication').'</button>';
$form->addAnything(__('Disable Two Factor Authentication'), $otp_html);
}
// edit mode messagge
if ($form->edit_mode) {
if (isset($rec_d['user_image'])) {
if ($imageDisk->isExists('persons/'.$rec_d['user_image'])) {
echo '<div id="memberImage" class="d-none"><img src="'.SWB.'lib/minigalnano/createthumb.php?filename=images/persons/'.urlencode($rec_d['user_image']).'&width=120&v='.date('his').'" alt="'.$rec_d['realname'].'" /></div>';
}
}
echo '<div class="per_title"><h2>'.__('Change User Profiles').'</h2></div>';
echo '<div class="infoBox row"><div class="col-6">'.__('You are going to edit user profile'),' : <b>'.$rec_d['realname'].'</b> <br />'.__('Last Update').' '.$rec_d['last_update'].'
<div>'.__('Leave Password field blank if you don\'t want to change the password').'</div></div>';
if ($rec_d['user_image']) {
if ($imageDisk->isExists('persons/'.$rec_d['user_image'])) {
echo '<div class="col-6 d-none"><div id="userImage" class="float-right"><img src="../images/persons/'.urlencode($rec_d['user_image']).'?'.date('this').'" class="w-100"/></div></div>';
}
}
echo '</div>';
}
// print out the form object
echo $form->printOut();
echo '<form id="formVerify2fa" target="blindSubmit" method="post" action="'.$_SERVER['PHP_SELF'].'?changecurrent=true"></form>';
?>
<div id="croppie-processor" style="visibility: hidden; position: absolute; width: 300px; height: 300px; top: -9999px;"></div>
<script type="text/javascript">
$(document).ready(function() {
const outputWidth = 160;
const outputHeight = 240;
const processorEl = $('#croppie-processor');
$('#image').on('change', function(){
const input = this;
$('#base64picstring').val('');
if (input.files && input.files[0]) {
let fileName = $(this).val().replace(/\\/g, '/').replace(/.*\//, '');
let fileExt = fileName.split('.').pop().toLowerCase();
$(this).parent('.custom-file').find('.custom-file-label').text(fileName);
// Check file extension
const allowedExt = <?php echo json_encode($sysconf['allowed_images']); ?>;
if (!allowedExt.includes('.' + fileExt)) {
alert('Tipe file tidak diizinkan. Gunakan: ' + allowedExt.join(', '));
$(this).val('');
$(this).parent('.custom-file').find('.custom-file-label').text('');
return;
}
const reader = new FileReader();
reader.onload = function (e) {
if (processorEl.data('croppie')) {
processorEl.croppie('destroy');
}
let $image_crop = processorEl.croppie({
enableExif: true,
enableZoom: true,
enableResize: false,
enableOrientation: true,
viewport: {
width: outputWidth,
height: outputHeight,
type: 'square'
},
boundary: {
width: outputWidth + 100,
height: outputHeight + 100
}
});
$image_crop.croppie('bind', {
url: e.target.result,
zoom: 0
}).then(function() {
setTimeout(function() {
// Get format based on file extension
let format = 'jpeg';
if (fileExt === 'png') format = 'png';
if (fileExt === 'gif') format = 'gif';
$image_crop.croppie('result', {
type: 'base64',
size: { width: outputWidth, height: outputHeight },
format: format,
quality: 0.9
}).then(function(base64_result) {
// Clean base64 string - remove data:image/xxx;base64, prefix if exists
let cleanBase64 = base64_result;
if (cleanBase64.includes('base64,')) {
cleanBase64 = cleanBase64.split('base64,')[1];
}
// Remove any whitespace or newlines
cleanBase64 = cleanBase64.replace(/\s/g, '');
// Append image type information
$('#base64picstring').val(cleanBase64 + '#image/type#' + format);
$('#current_image_preview').attr('src', base64_result);
$image_crop.croppie('destroy');
});
}, 50);
});
}
reader.readAsDataURL(input.files[0]);
}
});
$('.removeImage').click(function (e) {
if (confirm('Are you sure you want to permanently remove this image?')) {
$('#base64picstring').val('');
return true;
} else {
return false;
}
});
$(document).on('change', '.custom-file-input', function () {
let fileName = $(this).val().replace(/\\/g, '/').replace(/.*\//, '');
$(this).parent('.custom-file').find('.custom-file-label').text(fileName);
});
});
</script>
<?php
if ($sysconf['password_policy_strong']) {
echo simbio_security::validatePasswordFunctionJS();
echo '<script type="text/javascript">comparePassword("#mainForm", "#passwd1", "#passwd2", '.$sysconf['password_policy_min_length'].');</script>';
}
} else {
// only administrator have privileges to view user list
if (!($can_read AND $can_write) OR $_SESSION['uid'] != 1) {
die('<div class="errorBox">'.__('You don\'t have enough privileges to view this section').'</div>');
}
/* USER LIST */
// table spec
$table_spec = 'user AS u';
// create datagrid
$datagrid = new simbio_datagrid();
if ($can_read AND $can_write) {
$datagrid->setSQLColumn('u.user_id',
'u.realname AS \''.__('Real Name').'\'',
'u.username AS \''.__('Login Username').'\'',
'u.user_type AS \''.__('User Type').'\'',
'u.last_login AS \''.__('Last Login').'\'',
'u.last_update AS \''.__('Last Update').'\'');
$col = 3;
} else {
$datagrid->setSQLColumn('u.realname AS \''.__('Real Name').'\'',
'u.username AS \''.__('Real Name').'\'',
'u.user_type AS \''.__('User Type').'\'',
'u.last_login AS \''.__('Last Login').'\'',
'u.last_update AS \''.__('Last Update').'\'');
$col = 2;
}
$datagrid->modifyColumnContent($col, 'callback{getUserType}');
$datagrid->setSQLorder('username ASC');
// is there any search
$criteria = 'u.user_id != 1 ';
if (isset($_GET['keywords']) AND $_GET['keywords']) {
$keywords = $dbs->escape_string($_GET['keywords']);
$criteria .= " AND (u.username LIKE '%$keywords%' OR u.realname LIKE '%$keywords%')";
}
$datagrid->setSQLCriteria($criteria);
// set table and table header attributes
$datagrid->table_attr = 'id="dataList" class="s-table table"';
$datagrid->table_header_attr = 'class="dataListHeader" style="font-weight: bold;"';
// set delete proccess URL
$datagrid->chbox_form_URL = $_SERVER['PHP_SELF'];
// put the result into variables
$datagrid_result = $datagrid->createDataGrid($dbs, $table_spec, 20, ($can_read AND $can_write));
if (isset($_GET['keywords']) AND $_GET['keywords']) {
$keywords = htmlspecialchars($_GET['keywords'], ENT_QUOTES, 'UTF-8');
$msg = str_replace('{result->num_rows}', $datagrid->num_rows, __('Found <strong>{result->num_rows}</strong> from your keywords')); //mfc
echo '<div class="infoBox">'.$msg.' : "'.$keywords.'"</div>';
}
echo $datagrid_result;
}
/* main content end */
?>