A lot of changes

This commit is contained in:
WolverinDEV 2019-10-13 21:33:07 +02:00
parent e9384bcf18
commit 69d39991e4
41 changed files with 1576 additions and 749 deletions

View file

@ -1,5 +1,4 @@
<?php
$GLOBALS["COOKIE_NAME_USER_DATA"] = "user_data";
$GLOBALS["COOKIE_NAME_USER_SIGN"] = "user_sign";
@ -8,20 +7,13 @@
if($host == "WolverinDEV")
$localhost = true;
/*
openssl genrsa -des3 -out forum_private.pem 2048
openssl rsa -in forum_private.pem -outform PEM -pubout -out forum_public.pem
openssl rsa -in forum_private.pem -out private_unencrypted.pem -outform PEM #Export the private key as unencripted
*/
function authPath() {
if (file_exists("auth")) {
return "auth/";
} else return "";
}
function mainPath()
{
function mainPath() {
global $localhost;
if ($localhost) {
return "../";
@ -66,40 +58,6 @@
return ((int)$mt[1]) * 1000 + ((int)round($mt[0] * 1000));
}
/**
* @param $user \XF\Entity\User
* @return array
*/
function generateUserData($user)
{
$user_data = array();
$user_data["user_id"] = $user->user_id;
$user_data["user_name"] = $user->username;
$user_data["user_group"] = $user->user_group_id;
$user_data["user_groups"] = $user->secondary_group_ids;
$user_data["trophy_points"] = $user->trophy_points;
$user_data["register_date"] = $user->register_date;
$user_data["is_staff"] = $user->is_staff;
$user_data["is_admin"] = $user->is_admin;
$user_data["is_super_admin"] = $user->is_super_admin;
$user_data["is_banned"] = $user->is_banned;
$user_data["data_age"] = milliseconds();
$data = json_encode($user_data);
$file = realpath("./certs/private_unencrypted.pem");
$pkeyid = openssl_pkey_get_private("file://" . $file);
if (!$pkeyid) die("Could not open private key! Message: " . openssl_error_string() . " (" . $file . ")");
if (!openssl_sign($data, $signature, $pkeyid, OPENSSL_ALGO_SHA256)) die("Could not sign user data");
openssl_free_key($pkeyid);
return ["data" => $data, "sign" => base64_encode($signature)];
}
/**
* @param $username
* @param $password
@ -129,7 +87,7 @@
$user = $loginService->validate($password, $error);
if ($user) {
$response["success"] = true;
$allowed = true;
$allowed = false;
foreach ($allowedXFGroups as $id) {
foreach ($user->secondary_group_ids as $assigned)
if ($assigned == $id) {
@ -158,12 +116,6 @@
$response["sessionName"] = $session->getCookieName();
$response["sessionId"] = $session->getSessionId();
$response["user_name"] = $user->username;
$user_data = generateUserData($user);
$response["cookie_name_data"] = $GLOBALS["COOKIE_NAME_USER_DATA"];
$response["cookie_name_sign"] = $GLOBALS["COOKIE_NAME_USER_SIGN"];
$response["user_data"] = $user_data["data"];
$response["user_sign"] = $user_data["sign"];
} catch (Exception $error) {
$response["success"] = false;
$response["msg"] = $error->getMessage();
@ -199,10 +151,10 @@
}
/**
* @param null $sessionId
* @return int 0 = Success | 1 = Invalid coocie | 2 = invalid session
*/
function test_session($sessionId = null)
{
function test_session($sessionId = null) {
$app = getXF();
if(!$app) return -1;
@ -217,8 +169,7 @@
return 0;
}
function redirectOnInvalidSession()
{
function redirectOnInvalidSession() {
$app = getXF();
if(!$app) return;
@ -246,9 +197,11 @@
getXF(); /* Initialize XF */
}
if(!$_INCLIDE_ONLY) {
if(!defined("_AUTH_API_ONLY")) {
$app = getXF();
if(!$app) return;
if(!$app) {
die("failed to start app");
}
if (isset($_GET["type"])) {
error_log("Got authX request!");

View file

@ -1,8 +0,0 @@
<?php
/**
* Created by PhpStorm.
* User: wolverindev
* Date: 16.03.18
* Time: 17:03
*/
phpinfo();

View file

@ -1,5 +1,6 @@
<?php
include_once('auth.php');
$session = test_session();
if($session == 0) {
header('Location: ' . mainPath() . 'index.php');

View file

@ -991,46 +991,59 @@ $client_info_avatar_size: 10em;
align-self: center;
border-radius: 50%;
overflow: hidden;
.avatar {
overflow: hidden;
width: $client_info_avatar_size;
height: $client_info_avatar_size;
border-radius: 50%;
}
}
.container-avatar-edit {
position: absolute;
display: inline-block;
left: 0;
right: 0;
top: 0;
bottom: 0;
z-index: 2;
text-align: center;
> img {
cursor: pointer;
width: $client_info_avatar_size;
height: $client_info_avatar_size;
padding: calc(#{$client_info_avatar_size} / 6);
overflow: hidden;
opacity: 0;
&:hover {
opacity: .75;
}
@include transition(opacity $button_hover_animation_time ease-in-out);
}
.container-avatar-edit {
position: absolute;
display: none;
left: 0;
right: 0;
top: 0;
bottom: 0;
z-index: 2;
text-align: center;
> img {
cursor: pointer;
width: $client_info_avatar_size;
height: $client_info_avatar_size;
padding: calc(#{$client_info_avatar_size} / 6);
overflow: hidden;
opacity: 0;
&:hover {
opacity: 1;
}
@include transition(opacity $button_hover_animation_time ease-in-out);
}
}
&.editable {
.container-avatar-edit {
display: inline-block;
}
.container-avatar-edit:hover + .avatar {
opacity: .5;
}
}
}
.client-name {

View file

@ -1,12 +1,15 @@
@import "mixin";
$separator_thickness: 5px;
$small_device: 650px;
$animation_length: .5s;
.app {
min-width: 350px;
min-width: 600px;
min-height: 330px;
padding: 5px;
.container-app-main {
height: 100%;
width: 100%;
@ -67,6 +70,8 @@ $animation_length: .5s;
> .channel-tree {
padding-top: 5px;
min-height: 5em;
flex-grow: 1;
flex-shrink: 1;
}
@ -194,13 +199,7 @@ $animation_length: .5s;
top: 0;
overflow: auto;
padding: 5px;
}
@media only screen and (max-width: 400px), only screen and (max-height: 400px) {
.app-container {
overflow: auto;
}
padding: 0;
}
@media only screen and (max-width: $small_device) {
@ -220,11 +219,7 @@ $animation_length: .5s;
$animation_seperator_length: .1s;
.container-seperator {
-moz-transition: all $animation_seperator_length ease-in;
-o-transition: all $animation_seperator_length ease-in;
-webkit-transition: all $animation_seperator_length ease-in;
transition: all $animation_seperator_length ease-in;
@include transition(all $animation_seperator_length ease-in-out);
background: #1e1e1e;
flex-grow: 0;
@ -245,10 +240,7 @@ $animation_seperator_length: .1s;
}
&.seperator-selected {
-moz-transition: all $animation_seperator_length ease-in;
-o-transition: all $animation_seperator_length ease-in;
-webkit-transition: all $animation_seperator_length ease-in;
transition: all $animation_seperator_length ease-in;
@include transition(all $animation_seperator_length ease-in-out);
background-color: #707070;
}
@ -287,24 +279,4 @@ html, body {
body {
background: #1e1e1e !important;
}
footer {
position: fixed;
width: 100%;
bottom: 0px;
left: 0px;
right: 0px;
height: 25px;
background-color: lightgray;
display: flex;
}
footer .container {
width: 100%;
display: flex;
position: relative;
vertical-align: center;
justify-content: center;
}

View file

@ -5,6 +5,12 @@
text-align: center;
color: #999999;
.container-left {
display: flex;
flex-direction: column;
justify-content: center;
}
.container-right {
text-align: left;
padding-left: 2em;

View file

@ -1,8 +1,28 @@
@import "properties";
@import "mixin";
.modal-icon-select {
@include user-select(none);
display: flex;
flex-direction: column;
justify-content: stretch;
width: 50em;
/*
.right, .left {
.header {
text-transform: uppercase;
color: #557edc;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
}
*/
.container-icons {
flex-grow: 1;
flex-shrink: 1;
@ -41,11 +61,16 @@
overflow-x: hidden;
overflow-y: auto;
background-color: whitesmoke;
-webkit-user-select: none;
-moz-user-select: none;
-ms-user-select: none;
user-select: none;
background-color: $color_list_background;
border: 1px $color_list_border solid;
border-radius: $border_radius_large;
padding: .5em;
&.container-icons-local {
font-size: 16px;
}
.icon-container, .icon {
margin-left: 1px;
@ -57,11 +82,15 @@
cursor: pointer;
&:hover {
border-radius: .1em;
background-color: #00000011;
border: 1px solid black;
}
&.selected {
border-radius: .1em;
background-color: #00330011;
border: 1px solid red;
}
@ -82,8 +111,11 @@
left: 0;
right: 0;
font-size: 1.1em;
color: hsla(0, 0%, 40%, 1);
position: absolute;
background-color: grey;
background-color: #00000045;
cursor: not-allowed;
@ -118,16 +150,27 @@
display: flex;
flex-direction: row;
justify-content: stretch;
justify-content: space-between;
.spacer {
min-width: 0;
flex-grow: 1;
flex-shrink: 1;
}
.btn {
button {
flex-grow: 0;
flex-shrink: 0;
flex-shrink: 1;
width: 8em;
min-width: 4em;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
margin-right: 1em;
}
.button-select {
@ -136,10 +179,24 @@
display: flex;
align-items: center;
flex-direction: row;
justify-content: center;
.selected-item-container {
height: 16px;
vertical-align: sub;
a, div {
align-self: center;
}
> div {
font-size: 16px;
display: flex;
flex-direction: column;
justify-content: center;
margin-left: .5rem;
> div {
display: flex;
}
}
}
@ -150,17 +207,18 @@
}
.modal-icon-upload {
user-select: none;
-moz-user-select: none;
-ms-user-select: none;
-webkit-user-select: none;
@include user-select(none);
width: 50em;
min-width: 300px;
padding: 0!important;
display: flex;
flex-direction: column;
.container-select {
padding: 1em;
min-height: 130px;
display: flex;
@ -176,36 +234,44 @@
min-height: 130px;
overflow-y: auto;
margin-right: 5px;
margin-right: 1em;
border: gray solid 1px;
border-radius: 2px;
background-color: $color_list_background;
border: 1px $color_list_border solid;
border-radius: $border_radius_large;
padding: .5em;
display: block;
.icon-container {
display: inline-block;
height: 18px;
width: 18px;
image {
height: 16px;
width: 16px;
}
margin: 1px;
padding: 1px;
cursor: pointer;
&:hover {
padding: 0;
border-radius: .1em;
background-color: #00000011;
border: 1px solid black;
}
&.selected {
padding: 0;
border-radius: .1em;
background-color: #00330011;
border: 1px solid red;
}
&:hover, &.selected {
width: 18px;
height: 18px;
margin: -1px 0px;
}
> img {
height: 16px;
width: 16px;
}
}
}
@ -224,36 +290,67 @@
display: flex;
flex-direction: column;
justify-content: flex-start;
> button:not(:first-of-type) {
margin-top: .5em;
}
}
}
}
.container-upload {
-webkit-box-shadow: 0px -5px 5px 0px rgba(0,0,0,0.25);
-moz-box-shadow: 0px -5px 5px 0px rgba(0,0,0,0.25);
box-shadow: 0px -5px 2px 0px rgba(0, 0, 0, 0.25);
display: flex;
flex-direction: column;
margin-top: 5px;
border-top: 1px solid darkgray;
padding-top: 5px;
padding: .5em 1em 1em;
.container-error, .container-success {
width: 100%;
min-height: 60px;
display: inline-block;
.error-message, .message {
border: 2px solid;
border-radius: $border_radius_middle;
&.container-error {
border-color: #8000007F;
background-color: #80000040;
}
&.container-success {
margin-top: .5em;
border-color: #328f3340;
background-color: #328f3320;
}
padding: .5em;
display: flex;
flex-direction: row;
justify-content: space-between;
> * {
align-self: center;
display: inline-block;
}
button {
float: right;
display: inline-block;
width: 6em;
}
}
.container-success {
margin-top: 5px;
min-height: 100px;
&.hidden {
opacity: 0;
height: 0;
min-height: 0;
padding: 0;
margin: 0;
}
@include transition(.25s ease-in-out);
}
.container-info {
@ -263,13 +360,17 @@
}
.container-process {
margin-top: .5em;
width: 100%;
min-height: 100px;
overflow-y: auto;
border: gray solid 1px;
border-radius: 2px;
background-color: $color_list_background;
border: 1px $color_list_border solid;
border-radius: $border_radius_large;
.upload-entry {
display: flex;
@ -286,11 +387,13 @@
align-self: center;
display: flex;
flex-direction: column;
justify-content: center;
> img {
height: 16px;
width: 16px;
vertical-align: unset;
}
}
@ -300,19 +403,53 @@
flex-grow: 1;
flex-shrink: 1;
display: flex;
flex-direction: row;
justify-content: flex-start;
min-width: 2em;
margin: 2px 5px 2px 3px;
height: 16px;
overflow: hidden;
font-size: .75rem;
background-color: #222222;
border: 1px solid hsla(0, 0%, 10%, 1);
border-radius: .25rem;
.progress-bar {
height: 100%;
&.bg-danger {
background-color: #8000007F;
}
&.bg-success {
background-color: #328f337F;
}
@include transition(width 1s ease-in-out, background-color $button_hover_animation_time ease-in-out);
}
.progress-message {
align-self: center;
text-align: center;
position: absolute;
width: 100%;
display: flex;
flex-direction: row;
justify-content: center;
flex-grow: 1;
flex-shrink: 1;
min-width: 1em;
line-height: normal;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
}
}

View file

@ -1,25 +1,61 @@
@import "mixin";
@import "properties";
.modal-invite {
user-select: none;
padding: .5em!important;
@include user-select(none);
.general-properties {
flex: 0;
display: flex;
flex-direction: row;
flex-direction: column;
justify-content: stretch;
> .form-group:first-of-type {
flex-grow: 1;
margin-right: 5px;
.form-group {
flex: 0;
}
.container-settings {
flex: 0;
margin-bottom: .5em;
display: flex;
flex-direction: row;
justify-content: space-between;
label {
display: flex;
flex-direction: row;
justify-content: flex-start;
> * {
align-self: center;
}
a {
margin-left: .5em;
}
}
}
}
.text-output {
border-radius: 2px;
padding: 5px;
background: #00000012;
min-height: 120px;
background-color: $color_list_background;
border: 1px $color_list_border solid;
border-radius: $border_radius_large;
padding: .5em;
min-height: 5em;
width: 100%;
resize: none;
color: #999999;
@include user-select(text);
}
.buttons {

View file

@ -22,6 +22,7 @@
width: 100%;
max-height: 90vh;
min-height: 20em;
height: 100000000px; /* enforce max height */
}

View file

@ -1,3 +1,6 @@
@import "properties";
@import "mixin";
.query-create {
display: flex;
flex-direction: column;
@ -51,109 +54,303 @@
}
}
.query-management {
height: 100%;
.modal-body.modal-query-manage {
display: flex;
flex-direction: column;
flex-direction: row!important;
justify-content: stretch;
.header, .footer {
flex-grow: 0;
flex-shrink: 0;
}
padding: 1em!important;
min-width: 25em!important; /* 23em to be exact */
min-height: 20em!important;
width: 60em; /* recommend width */
height: 50em;
@include user-select(none);
.container {
flex-grow: 1;
flex-shrink: 1;
min-width: 20em;
min-height: 10em;
.header {
display: flex;
flex-direction: row;
justify-content: stretch;
.buttons {
flex-grow: 0;
}
.search {
margin-left: 5px;
flex-grow: 1;
input {
width: 100%;
}
}
}
.query-list {
margin-top: 5px;
.left, .right {
flex-grow: 1;
flex-shrink: 1;
display: flex;
flex-grow: 1;
flex-direction: column;
justify-content: stretch;
.column {
&.column-username {
width: calc(50% - 75px)
}
&.column-unique-id {
width: calc(50% - 75px)
}
&.column-bound-server {
width: 150px;
flex-grow: 0;
}
}
.query-list-header {
> .title {
flex-grow: 0;
flex-shrink: 0;
margin-bottom: .5em;
display: flex;
flex-direction: row;
justify-content: stretch;
.column {
border: 1px solid lightgray;
text-align: center;
a {
font-weight: bold;
color: #e0e0e0;
flex-grow: 1;
flex-shrink: 1;
font-size: 1.05em;
min-width: 5em;
align-self: flex-end;
line-height: normal;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
button {
flex-grow: 0;
flex-shrink: 1;
height: 2em;
font-size: .9em;
width: 10em;
min-width: 5em;
align-self: center;
overflow: hidden;
text-overflow: ellipsis;
}
}
}
.query-list-entries-container {
.left {
margin-right: .4em;
min-width: 10em;
.container-list {
flex-grow: 1;
display: flex;
flex-direction: column;
justify-content: start;
overflow-y: auto;
min-height: 250px;
justify-content: stretch;
.entry {
display: flex;
flex-direction: row;
border-radius: 0.2em;
border: 1px solid #1f2122;
background-color: #28292b;
.column {
margin-left: 2px;
.container-entries {
flex-shrink: 1;
flex-grow: 1;
min-height: 5em;
position: relative;
overflow-x: hidden;
overflow-y: auto;
@include chat-scrollbar-vertical();
.container-empty, .container-error {
position: absolute;
top: 0;
bottom: 0;
left: 0;
right: 0;
display: flex;
flex-direction: column;
justify-content: center;
text-align: center;
font-size: 2em;
background-color: #28292b;
color: hsla(0, 0%, 30%, 1);
}
cursor: pointer;
.container-error {
color: #732626;
}
&.selected {
background-color: blue;
.entry {
display: flex;
flex-direction: row;
justify-content: flex-start;
flex-shrink: 1;
min-width: 4em;
padding-left: .5em;
padding-right: .5em;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
cursor: pointer;
&:hover {
background-color: #2c2d2f;
}
&.selected {
background-color: #1a1a1b;
}
}
}
&.scrollbar {
.column-username {
width: calc(50% - 75px + 30px)
.container-search {
flex-shrink: 0;
flex-grow: 0;
padding: 0 .5em;
border-top: 1px solid #1f2122;
display: flex;
flex-direction: row;
justify-content: stretch;
button {
flex-grow: 0;
flex-shrink: 1;
height: 2em;
min-width: 2em;
align-self: center;
margin-top: .8em;
margin-right: .5em;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.column-unique-id {
width: calc(50% - 75px + 30px)
.form-group {
flex-grow: 1;
flex-shrink: 1;
min-width: 5em;
}
}
}
}
.footer {
margin-top: 5px;
display: flex;
flex-direction: row;
justify-content: space-between;
.right {
min-width: 10em;
margin-left: .4em;
.container-details {
flex-grow: 1;
display: flex;
flex-direction: column;
justify-content: stretch;
.detail {
flex-shrink: 0;
display: flex;
flex-direction: column;
justify-content: flex-start;
margin-bottom: 1em;
.title, .title a {
text-transform: uppercase;
color: #557edc;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.value {
@include user-select(text);
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
&.unique-id {
.title {
display: flex;
flex-direction: row;
justify-content: flex-start;
.button {
display: flex;
flex-direction: column;
justify-content: center;
width: 1.2em;
height: 1.2em;
align-self: center;
margin-left: .25em;
cursor: pointer;
border-radius: .2em;
&:hover {
background: #28292b;
}
margin-bottom: .2em; /* "text sub" */
> div {
align-self: center;
}
@include transition($button_hover_animation_time ease-in-out);
}
}
}
}
.spacer { flex-grow: 1; flex-shrink: 1; min-height: 0; }
.buttons {
display: flex;
flex-direction: row;
justify-content: space-between;
button {
flex-shrink: 1;
min-width: 5em;
height: 2em;
font-size: .9em;
width: 14em;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
&:not(:first-of-type) {
margin-left: 1em;
}
}
}
}
}
.container-seperator {
background: transparent;
}
}

View file

@ -103,7 +103,7 @@
flex-shrink: 1;
flex-grow: 1;
min-width: 25em;
min-width: 20em;
display: flex;
flex-direction: column;
@ -152,6 +152,15 @@
display: flex;
flex-direction: row;
justify-content: flex-start;
a {
flex-shrink: 1;
min-width: 0;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
}
}
}
@ -164,7 +173,7 @@
.container-button {
margin-right: 1em;
flex-shrink: 1;
flex-shrink: 1e8;
min-width: 5em;
display: flex;
@ -185,6 +194,8 @@
.right {
flex-grow: 1;
flex-shrink: 1;
min-width: 10em;
}
}
}

View file

@ -1,11 +1,6 @@
@import "properties";
@import "mixin";
$color_list_border: #161616;
$color_list_background: #28292b;
$color_list_hover: #2c2d2f;
$color_list_selected: #1a1a1b;
.modal-body.modal-settings {
padding: 0!important;
@ -338,6 +333,9 @@ $color_list_selected: #1a1a1b;
.header {
height: 3em;
flex-grow: 0;
flex-shrink: 0;
display: flex;
flex-direction: row;
justify-content: stretch;
@ -362,6 +360,9 @@ $color_list_selected: #1a1a1b;
flex-shrink: 0;
flex-grow: 0;
height: 2em;
align-self: flex-end;
margin-left: 1em;
min-width: 8em;
}
@ -466,6 +467,9 @@ $color_list_selected: #1a1a1b;
.left {
margin-right: 1em;
min-height: 0;
max-height: 100%;
.body {
flex-grow: 1;
flex-shrink: 1;

View file

@ -0,0 +1,49 @@
@import "mixin";
.modal-body.modal-volume {
@include user-select(none);
.info {
display: flex;
flex-direction: row;
justify-content: stretch;
.htmltag-client {
color: #999!important;
margin-left: .25em;
}
div {
display: flex;
flex-direction: row;
justify-content: stretch;
}
.value {
margin-left: .25em;
}
}
.container-slider {
margin-top: 1em;
}
.buttons {
display: flex;
flex-direction: row;
justify-content: stretch;
margin-top: 2em;
.spacer {
flex-grow: 1;
flex-shrink: 1;
min-width: 0;
}
button:not(:last-of-type) {
margin-right: 1em;
}
}
}

View file

@ -135,6 +135,9 @@
max-width: 100%;
min-width: 20em; /* may adjust if needed */
max-height: calc(100vh - 10em);
min-height: 5em;
overflow-y: auto;
overflow-x: auto;
}
@ -198,11 +201,12 @@
flex-grow: 0;
&.modal-header-error {
background-color: #ce0000;
//background-color: #ce0000;
background-color: hsla(0, 100%, 25%, 1);
}
&.modal-header-info {
background-color: #03a9f4;
background-color: hsla(199, 98%, 20%, 1);
}
&.modal-header-warning, &.modal-header-info, &.modal-header-error {
@ -307,6 +311,10 @@
}
}
}
.modal-body.modal-info, .modal-body.modal-error {
justify-content: center;
}
}
@ -346,6 +354,12 @@
line-height: 1;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
max-width: 100%;
&.bmd-label-floating {
will-change: left, top, contents;
color: #999999;

View file

@ -7,4 +7,9 @@ $border_radius_small: .1em;
$border_radius_middle: .15em;
$border_radius_large: .2em;
$button_hover_animation_time: .25s
$button_hover_animation_time: .25s;
$color_list_border: #161616;
$color_list_background: #28292b;
$color_list_hover: #2c2d2f;
$color_list_selected: #1a1a1b;

View file

@ -8,6 +8,20 @@
if(gethostname() == "WolverinDEV")
$localhost = true;
?>
<?php
if(!$localhost) {
/* Web Testing stuff */
define("_AUTH_API_ONLY", true);
if(file_exists("./auth.php"))
include "./auth.php";
else if(file_exists("./auth/auth.php"))
include "./auth/auth.php";
else
die("Missing auth handler");
redirectOnInvalidSession();
}
?>
<!DOCTYPE html>
<html lang="en">
<head>
@ -59,7 +73,7 @@
<meta name="format-detection" content="telephone=no">
<!-- Global site tag (gtag.js) - Google Analytics -->
<script async src="https://www.googletagmanager.com/gtag/js?id=UA-113151733-4"></script>
<script defer async src="https://www.googletagmanager.com/gtag/js?id=UA-113151733-4"></script>
<script>
window.dataLayer = window.dataLayer || [];
@ -121,32 +135,30 @@
</style>
</head>
<body>
<!-- No javascript error -->
<noscript>
<div class="fulloverlay no-js">
<div class="container">
<img src="img/script.svg" height="128px">
<h1>Please enable JavaScript</h1>
<h3>TeaSpeak web could not run without it!</h3>
<h3>Its like you, without coffee</h3>
</div>
</div>
</noscript>
<!-- loader setup -->
<div id="style">
<link rel="stylesheet" href="css/loader/loader.css">
</div>
<meta name="app-loader-target" content="app">
<div id="scripts">
<script type="application/javascript" src="loader/loader_app.min.js" defer></script>
<script type="application/javascript" src="loader/loader_app.js" defer></script>
<script type="application/javascript" src="loader/loader.js?_<?php echo time() ?>" defer></script>
<script type="application/javascript" src="loader/loader_app.min.js" async defer></script>
<script type="application/javascript" src="loader/loader_app.js" async defer></script>
<script type="application/javascript" src="loader/loader.js?_<?php echo time() ?>" async defer></script>
</div>
<!-- No javascript error -->
<div class="fulloverlay no-js">
<div class="container">
<img src="img/script.svg" height="128px">
<h1>Please enable JavaScript</h1>
<h3>TeaSpeak web could not run without it!</h3>
<h3>Its like you, without coffee</h3>
</div>
</div>
<script type="text/javascript" class="no-js">
let elements = document.getElementsByClassName("no-js");
while (elements.length > 0) //Removing these elements (even self)
elements.item(0).remove();
</script>
<!-- Loading screen -->
<div class="loader" id="loader-overlay">
<div class="half right"></div>
@ -173,7 +185,8 @@
</div>
</div>
<?php if($localhost && true) { ?>
<!-- debugging close -->
<?php if($localhost && false) { ?>
<div id="spoiler-style" style="z-index: 1000000; position: absolute; display: block; background: white; right: 5px; left: 5px; top: 34px;">
<!-- <img src="https://www.chromatic-solutions.de/teaspeak/window/connect_opened.png"> -->
<!-- <img src="http://puu.sh/DZDgO/9149c0a1aa.png"> -->
@ -218,7 +231,8 @@
<!-- <img src="http://puu.sh/E9jTe/b41f6386de.png"> -->
<!-- <img src="img/style/ban-list.png"> -->
<!-- <img src="http://puu.sh/E9jTe/b41f6386de.png"> -->
<img src="https://puu.sh/EhuVH/1e21540589.png">
<!-- <img src="https://puu.sh/EhuVH/1e21540589.png"> -->
<img src="https://puu.sh/EhvkJ/7551f548e3.png">
</div>
<button class="toggle-spoiler-style" style="height: 30px; width: 100px; z-index: 100000000; position: absolute; bottom: 2px;">toggle style</button>
<script>

View file

@ -147,10 +147,12 @@
</div>
</div>
<div class="divider"></div>
-->
<div class="divider"></div>
<div class="button button-subscribe-mode">
<div class="icon_em" title="{{tr 'Toggle channel subscribe mode' /}}"></div>
</div>
-->
<!-- the query button -->
<div class="button-dropdown btn_query" title="{{tr 'Show/hide server queries' /}}">
@ -2538,9 +2540,27 @@
</script>
<script class="jsrender-template" id="tmpl_change_volume" type="text/html">
<div style="display: flex; justify-content: center; vertical-align: center">
<input type="range" min="0" max="{{>max_volume}}" value="100" class="volume_slider" style="width: 100%">
<div class="display_volume" style="width: 60px; align-self: center; text-align: center">&plusmn;0 %
<div> <!-- for the renderer -->
<div class="info">
<div>{{tr "Change volume for client "/}} <node key="client"></node> :</div>
<a class="value">error: value</a>
</div>
<div class="container-slider">
<div class="filler" style="width: 30%"></div>
<div class="thumb container-tooltip" style="left: 30%">
<div class="tooltip">
<a>86%</a>
</div>
</div>
</div>
<div class="buttons">
<button class="btn btn-blue button-reset">{{tr "Reset" /}}</button>
<div class="spacer"></div>
{{if !local}}
<button class="btn btn-success button-apply">{{tr "Apply" /}}</button>
{{/if}}
<button class="btn btn-danger button-cancel">{{tr "Cancel" /}}</button>
<button class="btn btn-success button-save">{{tr "Save" /}}</button>
</div>
</div>
</script>
@ -3753,39 +3773,56 @@
</div>
</script>
<script class="jsrender-template" id="tmpl_invite" type="text/html">
<div class="modal-invite">
<div class="general-properties">
<div class="form-group property-type">
<label>{{tr "Link type:" /}}</label>
<select class="form-control">
<option value="0">{{tr "TeaWeb" /}}</option>
<option value="1" disabled>{{tr "TeaClient (Not supported yet)" /}}</option>
</select>
</div>
<div class="form-group">
<div class="switch flag-direct-connect">
<label>
<input type="checkbox">
{{tr "Connect directly" /}}
</label>
</div>
<div class="switch flag-resolved-address">
<label>
<input type="checkbox">
{{tr "Use resolved address" /}}
</label>
</div>
</div>
<div class="general-properties">
<div class="form-group property-type">
<label>{{tr "Link type:" /}}</label>
<select class="form-control">
<option value="tea-web">{{tr "TeaWeb" /}}</option>
<option value="tea-client">{{tr "TeaClient" /}}</option>
<option value="teamspeak">{{tr "TeamSpeak" /}}</option>
</select>
</div>
<textarea class="text-output" readonly></textarea>
<div class="buttons">
<button class="btn btn-primary button-copy">
<div class="icon client-copy"></div>
{{tr "Copy" /}}
</button>
<button class="btn btn-primary button-close">{{tr "Close" /}}</button>
<label class="container-settings">
<label>
<div class="checkbox flag-direct-connect">
<input type="checkbox">
<div class="mark"></div>
</div>
<a>{{tr "Connect directly" /}}</a>
<!--
<div class="help-tip tip-center tip-small">
<p>
{{tr "Lets the user directly connect to the server and not open the connect modal" /}}
</p>
</div>
-->
</label>
<label>
<div class="checkbox flag-resolved-address">
<input type="checkbox">
<div class="mark"></div>
</div>
<a>{{tr "Use resolved address" /}}</a>
<!--
<div class="help-tip tip-center tip-small">
<p>
{{tr "Use the resolved server address (IP) instead of the given address" /}}
</p>
</div>
-->
</label>
</div>
</div>
<textarea class="text-output" readonly></textarea>
<div class="buttons">
<button class="btn btn-primary button-copy">
<div class="icon client-copy"></div>
{{tr "Copy" /}}
</button>
<button class="btn btn-primary button-close">{{tr "Close" /}}</button>
</div>
</script>
<script class="jsrender-template" id="tmpl_query_create" type="text/html">
@ -3828,51 +3865,65 @@
</script>
<script class="jsrender-template" id="tmpl_query_manager" type="text/html">
<div class="query-management">
<div class="header">
<div class="form-group bmd-form-group buttons">
<button class="btn btn-success button button-query-create">{{tr "Create account" /}}</button>
<button class="btn btn-danger button button-query-delete">{{tr "Delete account" /}}</button>
<button class="btn btn-primary button button-query-rename">{{tr "Rename account" /}}</button>
<button class="btn btn-primary button button-query-change-password">{{tr "Change password" /}}
</button>
<div class="container"> <!-- required for the seperator -->
<div class="left">
<div class="title">
<a>{{tr "Account list" /}}</a>
<button class="btn btn-success button-create">{{tr "Create account" /}}</button>
</div>
<div class="form-group search">
<label class="bmd-label-floating">{{tr "search" /}}</label>
<input class="form-control input input-search" type="text">
</div>
</div>
<div class="query-list">
<div class="query-list-header">
<div class="column column-username">{{tr "Username" /}}</div>
<div class="column column-unique-id">{{tr "Unique ID" /}}</div>
<div class="column column-bound-server">{{tr "Bounded Server" /}}</div>
</div>
<div class="query-list-entries-container">
<div class="query-list-entries">
<div class="container-list">
<div class="container-entries">
<div class="container-empty">
error: empty
</div>
<div class="container-error">
error: query list
</div>
</div>
<div class="container-search">
<button class="btn btn-blue button-update">{{tr "Refresh" /}}</button>
<div class="form-group">
<label class="bmd-label-floating">{{tr "Search for query" /}}</label>
<input type="text" class="form-control filter-input">
</div>
</div>
</div>
</div>
<div class="footer">
<div class="info">
<a>{{tr "loading..." /}}</a>
<div class="container-seperator vertical" seperator-id="seperator-query-manage"></div>
<div class="right">
<div class="title">
<a>{{tr "Account details" /}}</a>
<button class="btn btn-danger button-delete">{{tr "Delete account" /}}</button>
</div>
<div class="buttons">
<button class="btn btn-primary button-refresh">{{tr "Refresh" /}}</button>
<div class="container-details">
<div class="detail login-name">
<a class="title">{{tr "Login name" /}}</a>
<a class="value">error: login name</a>
</div>
<div class="detail unique-id">
<div class="title">
<a>{{tr "Unique ID" /}}</a>
<div class="button button-copy">
<div class="icon_em client-copy"></div>
</div>
</div>
<a class="value">error: unique id</a>
</div>
<div class="detail bound-server">
<a class="title">{{tr "Bound server ID" /}}</a>
<a class="value">error: bound server</a>
</div>
<div class="spacer"></div>
<div class="buttons">
<button class="btn btn-blue button-change-password">{{tr "Change password" /}}</button>
<button class="btn btn-blue button-rename">{{tr "Rename Account" /}}</button>
</div>
</div>
</div>
</div>
</script>
<script class="jsrender-template" id="tmpl_query_manager-list_entry" type="text/html">
<div class="entry">
<div class="column column-username">{{>username}}</div>
<div class="column column-unique-id">{{>unique_id}}</div>
<div class="column column-bound-server">{{>bounded_server}}</div>
</div>
</script>
<script class="jsrender-template" id="tmpl_playlist_list" type="text/html">
<div class="playlist-management">
<div class="header">
@ -4345,87 +4396,83 @@
</script>
<script class="jsrender-template" id="tmpl_icon_select" type="text/html">
<div class="modal-icon-select">
<div class="container-icons">
<div class="group_box">
<div class="header">{{tr "Remote" /}}</div>
<div class="content">
<div class="container-icons-list">
<div class="container-icons-remote {{if enable_select || enable_delete}}icon-select{{/if}}"></div>
<div class="container-loading">
<a>{{tr "loading..." /}}</a>
</div>
<div class="container-no-permissions">
<a>{{tr "You dont have permissions the view the icons" /}}</a>
</div>
<div class="container-error">
<a class="error-message">{{ŧr "An error occured" /}}</a>
</div>
<div class="container-icons">
<div class="left">
<div class="header">{{tr "Remote" /}}</div>
<div class="content">
<div class="container-icons-list">
<div class="container-icons-remote {{if enable_select || enable_delete}}icon-select{{/if}}"></div>
<div class="container-loading">
<a>{{tr "loading..." /}}</a>
</div>
<div class="container-buttons">
{{if enable_upload}}
<button class="btn btn-success button-upload">{{tr "Upload" /}}</button>
{{/if}}
{{if enable_delete}}
<button class="btn btn-danger button-delete">{{tr "Delete" /}}</button>
{{/if}}
<div class="container-no-permissions">
<a>{{tr "You dont have permissions the view the icons" /}}</a>
</div>
<div class="container-error">
<a class="error-message">{{ŧr "An error occurred" /}}</a>
</div>
</div>
</div>
<div class="group_box">
<div class="header">{{tr "Local" /}}</div>
<div class="content">
<div class="container-icons-list">
<div class="container-icons-local {{if enable_select}}icon-select{{/if}}"></div>
</div>
<div class="container-buttons">
{{if enable_upload}}
<button class="btn btn-success button-upload">{{tr "Upload" /}}</button>
{{/if}}
{{if enable_delete}}
<button class="btn btn-danger button-delete">{{tr "Delete" /}}</button>
{{/if}}
</div>
</div>
</div>
<div class="container-buttons">
<button class="btn btn-primary btn-raised button-reload">{{tr "Reload" /}}</button>
<div class="spacer"></div>
{{if enable_select}}
<button class="btn btn-success btn-raised button-select-no-icon">{{tr "Remove icon" /}}</button>
<button class="btn btn-success btn-raised button-select"><a>{{tr "Select " /}}</a>
<div class="selected-item-container"></div>
</button>
{{/if}}
<div class="right">
<div class="header">{{tr "Local" /}}</div>
<div class="content">
<div class="container-icons-list">
<div class="container-icons-local {{if enable_select}}icon-select{{/if}}"></div>
</div>
</div>
</div>
</div>
<div class="container-buttons">
<button class="btn btn-primary btn-raised button-reload">{{tr "Reload" /}}</button>
<div class="spacer"></div>
{{if enable_select}}
<button class="btn btn-success btn-raised button-select-no-icon">{{tr "Remove icon" /}}</button>
<button class="btn btn-success btn-raised button-select"><a>{{tr "Select " /}}</a>
<div class="selected-item-container"></div>
</button>
{{/if}}
</div>
</script>
<script class="jsrender-template" id="tmpl_icon_upload" type="text/html">
<div class="modal-icon-upload">
<div class="container-select">
<div class="container-icons"></div>
<div class="container-buttons">
<div class="buttons-manage">
<button class="btn btn-primary btn-raised button-add">{{tr "Add icon" /}}</button>
<button class="btn btn-danger button-remove">{{tr "Remove selected" /}}</button>
</div>
<button class="btn btn-primary btn-raised button-upload"></button>
<div class="container-select">
<div class="container-icons"></div>
<div class="container-buttons">
<div class="buttons-manage">
<button class="btn btn-primary btn-raised button-add">{{tr "Add icon" /}}</button>
<button class="btn btn-danger button-remove">{{tr "Remove selected" /}}</button>
</div>
<button class="btn btn-primary btn-raised button-upload"></button>
<input type="file" class="input-file-upload" multiple/>
</div>
<input type="file" class="input-file-upload" accept="image/*" multiple/>
</div>
<div class="container-upload">
<div class="container-error alert alert-danger">
<div class="error-message">You're not connected. Failed to upload icons</div>
<button type="button" class="btn btn-danger btn-raised button-upload-abort">{{tr "abort" /}}
</button>
</div>
<div class="container-process"></div>
<div class="container-info">
<div class="container-info-uploaded">{{tr "Uploaded icons (total | successfully | error): "
/}}
</div>
<div class="uploaded-statistics"></div>
</div>
<div class="container-success alert alert-success">
<div class="message">Uploaded 10 icons successfully</div>
<button type="button" class="btn btn-success btn-raised button-upload-abort">{{tr "okey" /}}
</button>
</div>
<div class="container-upload">
<div class="container-error">
<div class="error-message">You're not connected. Failed to upload icons</div>
<button type="button" class="btn btn-danger btn-raised button-upload-abort">{{tr "abort" /}}
</button>
</div>
<div class="container-process"></div>
<div class="container-info">
<div class="container-info-uploaded">{{tr "Uploaded icons (total | successfully | error): "
/}}
</div>
<div class="uploaded-statistics"></div>
</div>
<div class="container-success">
<div class="message">Uploaded 10 icons successfully</div>
<button type="button" class="btn btn-success btn-raised button-upload-abort">{{tr "okey" /}}
</button>
</div>
</div>
</script>
@ -4573,18 +4620,19 @@
{{/if}}
<h2>{{tr "Special thanks" /}}</h2>
<p>
"Яedeemer" (Janni K.)
"Яedeemer" (Janni K.)<br>
Chromatic-Solutions (Sofian) for the lovely dark design
</p>
<h2>{{tr "Contact" /}}</h2>
<p>
{{tr "E-Mail:" /}} {{if client}}client{{else}}web{{/if}}@teaspeak.de<br>
{{tr "WWW:" /}} <a href="https://teaspeak.de">https://teaspeak.de</a><br>
{{tr "Community:" /}} <a href="https://teaspeak.de">https://forum.teaspeak.de</a>
{{tr "E-Mail:" /}} <a href="mailto:{{if client}}client{{else}}web{{/if}}.support@teaspeak.de">{{if client}}client{{else}}web{{/if}}.support@teaspeak.de</a><br>
{{tr "WWW:" /}} <a href="https://teaspeak.de" target="_blank">https://teaspeak.de</a><br>
{{tr "Community:" /}} <a href="https://teaspeak.de" target="_blank">https://forum.teaspeak.de</a>
</p>
<h2>{{tr "License" /}}</h2>
<p>
The {{if client}}TeaClient{{else}}TeaWeb{{/if}} application is licensed by MPL-2.0<br>
More information here: https://github.com/TeaSpeak/TeaWeb/blob/master/LICENSE.TXT
More information here: <a href="https://github.com/TeaSpeak/TeaWeb/blob/master/LICENSE.TXT" target="_blank">https://github.com/TeaSpeak/TeaWeb/blob/master/LICENSE.TXT</a>
</p>
</div>
</script>

View file

@ -615,7 +615,7 @@ class ConnectionHandler {
targetChannel = targetChannel || this.getClient().currentChannel();
const vconnection = this.serverConnection.voice_connection();
const basic_voice_support = this.serverConnection.support_voice() && vconnection.connected();
const basic_voice_support = this.serverConnection.support_voice() && vconnection.connected() && targetChannel;
const support_record = basic_voice_support && (!targetChannel || vconnection.encoding_supported(targetChannel.properties.channel_codec));
const support_playback = basic_voice_support && (!targetChannel || vconnection.decoding_supported(targetChannel.properties.channel_codec));

View file

@ -236,7 +236,9 @@ namespace bookmarks {
export function add_current_server() {
const ch = server_connections.active_connection_handler();
if(ch && ch.connected) {
createInputModal(tr("Enter bookmarks name"), tr("Please enter the bookmarks name:<br>"), text => true, result => {
const ce = ch.getClient();
const name = ce ? ce.clientNickName() : undefined;
createInputModal(tr("Enter bookmarks name"), tr("Please enter the bookmarks name:<br>"), text => text.length > 0, result => {
if(result) {
const bookmark = create_bookmark(result as string, bookmarks(), {
server_port: ch.serverConnection.remote_address().port,
@ -244,11 +246,13 @@ namespace bookmarks {
server_password: "",
server_password_hash: ""
}, this.connection_handler.getClient().clientNickName());
}, name);
save_bookmark(bookmark);
control_bar.update_bookmarks();
top_menu.rebuild_bookmarks();
createInfoModal(tr("Server added"), tr("Server has been successfully added to your bookmarks.")).open();
}
}).open();
} else {

View file

@ -114,7 +114,7 @@ namespace connection {
for(const entry of json) {
const rentry = {} as QueryListEntry;
rentry.bounded_server = entry["client_bounded_server"];
rentry.bounded_server = parseInt(entry["client_bound_server"]);
rentry.username = entry["client_login_name"];
rentry.unique_id = entry["client_unique_identifier"];

View file

@ -295,6 +295,17 @@ class Settings extends StaticSettings {
key: "font_size"
};
static readonly KEY_LAST_INVITE_LINK_TYPE: SettingsKey<string> = {
key: "last_invite_link_type",
default_value: "tea-web"
};
static readonly FN_INVITE_LINK_SETTING: (name: string) => SettingsKey<string> = name => {
return {
key: 'invite_link_setting_' + name
}
};
static readonly FN_SERVER_CHANNEL_SUBSCRIBE_MODE: (channel_id: number) => SettingsKey<number> = channel => {
return {
key: 'channel_subscribe_mode_' + channel

View file

@ -146,7 +146,7 @@ class ClientEntry {
this._tag = undefined;
}
if(this._audio_handle) {
console.warn(tr("Destroying client with an active audio handle. This could cause memory leaks!"));
log.warn(LogCategory.AUDIO, tr("Destroying client with an active audio handle. This could cause memory leaks!"));
this._audio_handle.abort_replay();
this._audio_handle.callback_playback = undefined;
this._audio_handle.callback_stopped = undefined;
@ -235,7 +235,7 @@ class ClientEntry {
}
}
protected initializeListener(){
protected initializeListener() {
if(this._listener_initialized) return;
this._listener_initialized = true;
@ -609,7 +609,7 @@ class ClientEntry {
icon_class: "client-volume",
name: tr("Change Volume"),
callback: () => {
Modals.spawnChangeVolume(this._audio_volume, volume => {
Modals.spawnChangeVolume(this, true, this._audio_volume, undefined, volume => {
this._audio_volume = volume;
this.channelTree.client.settings.changeServer("volume_client_" + this.clientUid(), volume);
if(this._audio_handle)
@ -1140,12 +1140,14 @@ class LocalClientEntry extends ClientEntry {
}
initializeListener(): void {
if(this._listener_initialized)
this.tag.off();
this._listener_initialized = false; /* could there be a better system */
super.initializeListener();
this.tag.find(".client-name").addClass("client-name-own");
this.tag.dblclick(() => {
if($.isArray(this.channelTree.currently_selected)) { //Multiselect
this.tag.on('dblclick', () => {
if(Array.isArray(this.channelTree.currently_selected)) { //Multiselect
return;
}
this.openRename();
@ -1153,8 +1155,6 @@ class LocalClientEntry extends ClientEntry {
}
openRename() : void {
const _self = this;
this.channelTree.client_mover.enabled = false;
const elm = this.tag.find(".client-name");
@ -1162,30 +1162,30 @@ class LocalClientEntry extends ClientEntry {
elm.removeClass("client-name-own");
elm.css("background-color", "white");
elm.focus();
_self.renaming = true;
this.renaming = true;
elm.keypress(function (e) {
if(e.keyCode == KeyCode.KEY_RETURN) {
$(this).trigger("focusout");
elm.on('keypress', event => {
if(event.keyCode == KeyCode.KEY_RETURN) {
$(event.target).trigger("focusout");
return false;
}
});
elm.focusout(e => {
elm.on('focusout', event => {
this.channelTree.client_mover.enabled = true;
if(!_self.renaming) return;
_self.renaming = false;
if(!this.renaming) return;
this.renaming = false;
elm.css("background-color", "");
elm.removeAttr("contenteditable");
elm.addClass("client-name-own");
let text = elm.text().toString();
if(_self.clientNickName() == text) return;
if(this.clientNickName() == text) return;
elm.text(_self.clientNickName());
const old_name = _self.clientNickName();
_self.handle.serverConnection.command_helper.updateClient("client_nickname", text).then((e) => {
elm.text(this.clientNickName());
const old_name = this.clientNickName();
this.handle.serverConnection.command_helper.updateClient("client_nickname", text).then((e) => {
settings.changeGlobal(Settings.KEY_CONNECT_USERNAME, text);
this.channelTree.client.log.log(log.server.Type.CLIENT_NICKNAME_CHANGED, {
client: this.log_data(),
@ -1197,7 +1197,7 @@ class LocalClientEntry extends ClientEntry {
this.channelTree.client.log.log(log.server.Type.CLIENT_NICKNAME_CHANGE_FAILED, {
reason: e.extra_message
});
_self.openRename();
this.openRename();
});
});
}
@ -1257,7 +1257,9 @@ class MusicClientEntry extends ClientEntry {
let trigger_close = true;
contextmenu.spawn_context_menu(x, y,
...this.contextmenu_info(), {
name: tr("<b>Change bot name</b>"),
name: (contextmenu.get_provider().html_format_enabled() ? "<b>" : "") +
tr("Change bot name") +
(contextmenu.get_provider().html_format_enabled() ? "</b>" : ""),
icon_class: "client-change_nickname",
disabled: false,
callback: () => {
@ -1269,7 +1271,7 @@ class MusicClientEntry extends ClientEntry {
});
}
}, { width: 400, maxLength: 255 }).open();
}, { width: "40em", min_width: "10em", maxLength: 255 }).open();
},
type: contextmenu.MenuEntryType.ENTRY
}, {
@ -1285,7 +1287,7 @@ class MusicClientEntry extends ClientEntry {
});
}
}, { width: 400, maxLength: 255 }).open();
}, { width: "60em", min_width: "10em", maxLength: 255 }).open();
},
type: contextmenu.MenuEntryType.ENTRY
},
@ -1375,7 +1377,7 @@ class MusicClientEntry extends ClientEntry {
icon_class: "client-volume",
name: tr("Change local volume"),
callback: () => {
Modals.spawnChangeVolume(this._audio_handle.get_volume(), volume => {
Modals.spawnChangeVolume(this, true, this._audio_handle.get_volume(), undefined, volume => {
this.channelTree.client.settings.changeServer("volume_client_" + this.clientUid(), volume);
this._audio_handle.set_volume(volume);
});
@ -1390,7 +1392,7 @@ class MusicClientEntry extends ClientEntry {
if(max_volume < 0)
max_volume = 100;
Modals.spawnChangeRemoteVolume(this.properties.player_volume, max_volume / 100, value => {
Modals.spawnChangeVolume(this, false, this.properties.player_volume, max_volume / 100, value => {
if(typeof(value) !== "number")
return;

View file

@ -62,7 +62,7 @@ class ModalProperties {
} else this.closeListener = listener;
return this;
}
width: number | string = "60%";
width: number | string;
min_width?: number | string;
height: number | string = "auto";
@ -147,9 +147,13 @@ class Modal {
Object.assign(properties, this.properties.template_properties);
const tag = template.renderTag(properties);
if(typeof(this.properties.width) !== "undefined")
if(typeof(this.properties.width) !== "undefined" && typeof(this.properties.min_width) !== "undefined")
tag.find(".modal-content")
.css("min-width", this.properties.min_width)
.css("width", this.properties.width);
else if(typeof(this.properties.width) !== "undefined") //Legacy support
tag.find(".modal-content").css("min-width", this.properties.width);
if(typeof(this.properties.min_width) !== "undefined")
else if(typeof(this.properties.min_width) !== "undefined")
tag.find(".modal-content").css("min-width", this.properties.min_width);
this.close_elements = tag.find(".button-modal-close");
@ -292,7 +296,10 @@ function createErrorModal(header: BodyCreator, message: BodyCreator, props: Moda
props.header = header;
props.body = message;
return createModal(props);
const modal = createModal(props);
modal.htmlTag.find(".modal-body").addClass("modal-error");
return modal;
}
function createInfoModal(header: BodyCreator, message: BodyCreator, props: ModalProperties | any = { footer: undefined }) {
@ -302,7 +309,9 @@ function createInfoModal(header: BodyCreator, message: BodyCreator, props: Modal
props.header = header;
props.body = message;
return createModal(props);
const modal = createModal(props);
modal.htmlTag.find(".modal-body").addClass("modal-info");
return modal;
}
/* extend jquery */

View file

@ -467,8 +467,8 @@ namespace top_menu {
item.click(() => {
const scon = server_connections.active_connection_handler();
if(scon && scon.connected) {
if(scon.permissions.neededPermission(PermissionType.B_CLIENT_CREATE_MODIFY_SERVERQUERY_LOGIN).granted(1)) {
Modals.spawnQueryManage(scon);
if(scon.permissions.neededPermission(PermissionType.B_CLIENT_CREATE_MODIFY_SERVERQUERY_LOGIN).granted(1) || scon.permissions.neededPermission(PermissionType.B_CLIENT_QUERY_CREATE).granted(1)) {
Modals.spawnQueryCreate(scon);
} else {
createErrorModal(tr("You dont have the permission"), tr("You dont have the permission to create a server query login")).open();
scon.sound.play(Sound.ERROR_INSUFFICIENT_PERMISSIONS);

View file

@ -2430,8 +2430,7 @@ test
else
this.handle.handle.fileManager.avatars.generate_chat_tag(undefined, undefined).appendTo(container_avatar);
const container_avatar_edit = this._html_tag.find(".container-avatar-edit");
container_avatar_edit.toggle(client instanceof LocalClientEntry);
container_avatar.toggleClass("editable", client instanceof LocalClientEntry);
}
/* updating the info fields */
{

View file

@ -1,3 +1,8 @@
/// <reference path="../../ConnectionHandler.ts" />
/// <reference path="../../ui/elements/modal.ts" />
/// <reference path="../../i18n/localize.ts" />
/// <reference path="../../proto.ts" />
namespace Modals {
export function openBanList(client: ConnectionHandler) {
let modal: Modal;
@ -413,7 +418,11 @@ namespace Modals {
return false;
}
tag.show().toggleClass("highlight", highlight_own);
tag.show().toggleClass(
"highlight",
highlight_own &&
entry.flag_own
);
return true;
});

View file

@ -3,145 +3,75 @@
/// <reference path="../../proto.ts" />
namespace Modals {
export function spawnChangeVolume(current: number, callback: (number) => void) {
let updateCallback: (number) => void;
const connectModal = createModal({
header: function() {
let header = $.spawn("div");
header.text(tr("Change volume"));
return header;
},
//TODO: Use the max limit!
let modal: Modal;
export function spawnChangeVolume(client: ClientEntry, local: boolean, current: number, max: number | undefined, callback: (number) => void) {
if(modal) modal.close();
let new_value: number;
modal = createModal({
header: local ? tr("Change local volume") : tr("Change remote volume"),
body: function () {
let tag = $("#tmpl_change_volume").renderTag({
max_volume: 200
client: htmltags.generate_client_object({
add_braces: false,
client_name: client.clientNickName(),
client_unique_id: client.properties.client_unique_identifier,
client_id: client.clientId()
}),
local: local
});
tag.find(".volume_slider").on("change",_ => updateCallback(tag.find(".volume_slider").val()));
tag.find(".volume_slider").on("input",_ => updateCallback(tag.find(".volume_slider").val()));
//connect_address
return tag;
},
footer: function () {
let tag = $.spawn("div");
tag.css("text-align", "right");
tag.css("margin-top", "3px");
tag.css("margin-bottom", "6px");
tag.addClass("modal-button-group");
let buttonReset = $.spawn("button");
buttonReset.text(tr("Reset"));
buttonReset.on("click", function () {
updateCallback(100);
});
tag.append(buttonReset);
let buttonCancel = $.spawn("button");
buttonCancel.text(tr("Cancel"));
buttonCancel.on("click", function () {
updateCallback(current * 100);
connectModal.close();
});
tag.append(buttonCancel);
let buttonOk = $.spawn("button");
buttonOk.text(tr("OK"));
buttonOk.on("click", function () {
connectModal.close();
});
tag.append(buttonOk);
return tag;
const container_value = tag.find(".info .value");
const set_value = value => {
const number = value > 100 ? value - 100 : 100 - value;
container_value.html((value == 100 ? "&plusmn;" : value > 100 ? "+" : "-") + number + "%");
new_value = value / 100;
if(local) callback(new_value);
};
set_value(current * 100);
const slider_tag = tag.find(".container-slider");
const slider = sliderfy(slider_tag, {
initial_value: current * 100,
step: 1,
max_value: 200,
min_value: 0,
unit: '%'
});
slider_tag.on('change', event => set_value(parseInt(slider_tag.attr("value"))));
tag.find(".button-save").on('click', event => {
if(typeof(new_value) !== "undefined") callback(new_value);
modal.close();
});
tag.find(".button-cancel").on('click', event => {
callback(current);
modal.close();
});
tag.find(".button-reset").on('click', event => {
slider.value(100);
});
tag.find(".button-apply").on('click', event => {
callback(new_value);
new_value = undefined;
});
return tag.children();
},
footer: null,
width: 600
});
updateCallback = value => {
connectModal.htmlTag.find(".volume_slider").val(value);
let display = connectModal.htmlTag.find(".display_volume");
let number = (value - 100);
display.html((number == 0 ? "&plusmn;" : number > 0 ? "+" : "") + number + " %");
callback(value / 100);
};
connectModal.open();
updateCallback(current * 100);
}
/* Units are between 0 and 1 */
export function spawnChangeRemoteVolume(current: number, max_value: number, callback: (value: number) => void) {
let update_volume: (number) => void;
let current_value = current; /* between 0 and 100! */
const modal = createModal({
header: function() {
let header = $.spawn("div");
header.text(tr("Change volume"));
return header;
},
body: function () {
let tag = $("#tmpl_change_volume").renderTag({
max_volume: Math.ceil(max_value * 100)
});
tag.find(".volume_slider").on("change",_ => update_volume(tag.find(".volume_slider").val()));
tag.find(".volume_slider").on("input",_ => update_volume(tag.find(".volume_slider").val()));
//connect_address
return tag;
},
footer: function () {
let tag = $.spawn("div");
tag.css("text-align", "right");
tag.css("margin-top", "3px");
tag.css("margin-bottom", "6px");
tag.addClass("modal-button-group");
{
let button_apply = $.spawn("button");
button_apply.text(tr("Apply"));
button_apply.on("click", () => {
callback(current_value / 100);
});
tag.append(button_apply);
}
{
let button_reset = $.spawn("button");
button_reset.text(tr("Reset"));
button_reset.on("click", () => update_volume(max_value * 100));
tag.append(button_reset);
}
{
let button_cancel = $.spawn("button");
button_cancel.text(tr("Cancel"));
button_cancel.on("click", () => modal.close());
tag.append(button_cancel);
}
{
let button_ok = $.spawn("button");
button_ok.text(tr("OK"));
button_ok.on("click", () => {
callback(current_value / 100);
modal.close();
});
tag.append(button_ok);
}
return tag;
},
width: 600
});
update_volume = value => {
modal.htmlTag.find(".volume_slider").val(value);
const tag_display = modal.htmlTag.find(".display_volume");
tag_display.html(value + " %");
current_value = value;
};
modal.close_listener.push(() => modal = undefined);
modal.open();
update_volume(current * 100);
modal.htmlTag.find(".modal-body").addClass("modal-volume");
}
}

View file

@ -11,17 +11,19 @@ namespace Modals {
header: tr("Icons"),
footer: undefined,
body: () => {
const template = $("#tmpl_icon_select").renderTag({
return $("#tmpl_icon_select").renderTag({
enable_select: !!callback_icon,
enable_upload: allow_manage,
enable_delete: allow_manage
});
},
return template;
}
min_width: "20em"
});
modal.htmlTag.find(".modal-body").addClass("modal-icon-select");
const button_select = modal.htmlTag.find(".button-select");
const button_delete = modal.htmlTag.find(".button-delete").prop("disabled", true);
const button_upload = modal.htmlTag.find(".button-upload").prop("disabled", !allow_manage);
@ -29,11 +31,15 @@ namespace Modals {
const container_loading = modal.htmlTag.find(".container-loading").hide();
const container_no_permissions = modal.htmlTag.find(".container-no-permissions").hide();
const container_error = modal.htmlTag.find(".container-error").hide();
const selected_container = modal.htmlTag.find(".selected-item-container");
const container_icons = modal.htmlTag.find(".container-icons");
const container_icons_remote = container_icons.find(".container-icons-remote");
const container_icons_local = container_icons.find(".container-icons-local");
const update_local_icons = (icons: number[]) => {
const container_icons = modal.htmlTag.find(".container-icons .container-icons-local");
container_icons.empty();
container_icons_local.empty();
for(const icon_id of icons) {
const tag = client.fileManager.icons.generateTag(icon_id, {animate: false}).attr('title', "Icon " + icon_id);
@ -53,7 +59,7 @@ namespace Modals {
if(icon_id == selected_icon)
tag.trigger('click');
}
tag.appendTo(container_icons);
tag.appendTo(container_icons_local);
}
};
@ -71,8 +77,6 @@ namespace Modals {
};
client.fileManager.requestFileList("/icons").then(icons => {
const container_icons = modal.htmlTag.find(".container-icons");
const container_icons_remote = container_icons.find(".container-icons-remote");
const container_icons_remote_parent = container_icons_remote.parent();
container_icons_remote.detach().empty();
@ -90,7 +94,7 @@ namespace Modals {
for(const icon of chunk) {
const icon_id = parseInt(icon.name.substr("icon_".length));
if(icon_id == NaN) {
if(Number.isNaN(icon_id)) {
log.warn(LogCategory.GENERAL, tr("Received an unparsable icon within icon list (%o)"), icon);
continue;
}
@ -270,13 +274,13 @@ namespace Modals {
} catch(error) {
console.log("Image failed to load (%o)", error);
console.error(tr("Failed to load file %s: Image failed to load"), file.name);
createErrorModal(tr("Icon upload failed"), tra("Failed to upload icon {}.<br>Failed to load image", file.name)).open();
createErrorModal(tr("Icon upload failed"), tra("Failed to upload icon {}.{:br:}Failed to load image", file.name)).open();
icon.state = "error";
}
const width_error = message => {
console.error(tr("Failed to load file %s: Invalid bounds: %s"), file.name, message);
createErrorModal(tr("Icon upload failed"), tra("Failed to upload icon {}.<br>Image is too large ({})", file.name, message)).open();
createErrorModal(tr("Icon upload failed"), tra("Failed to upload icon {}.{:br:}Image is too large ({})", file.name, message)).open();
icon.state = "error";
};
@ -376,6 +380,9 @@ namespace Modals {
let upload_key: transfer.UploadKey;
try {
await new Promise(resolve => setTimeout(resolve, 1000));
throw "test error";;
upload_key = await client.fileManager.upload_file({
channel: undefined,
channel_password: undefined,
@ -431,12 +438,12 @@ namespace Modals {
const modal = createModal({
header: tr("Upload Icons"),
footer: undefined,
body: () => {
const template = $("#tmpl_icon_upload").renderTag();
return template;
},
closeable: false
body: () => $("#tmpl_icon_upload").renderTag(),
closeable: false,
min_width: "20em"
});
modal.htmlTag.find(".modal-body").addClass("modal-icon-upload");
const button_upload = modal.htmlTag.find(".button-upload");
const button_delete = modal.htmlTag.find(".button-remove").prop("disabled", true);
@ -548,7 +555,7 @@ namespace Modals {
const show_critical_error = message => {
container_error.find(".error-message").text(message);
container_error.show();
container_error.removeClass("hidden");
};
const finish_upload = () => {
@ -563,7 +570,8 @@ namespace Modals {
button_upload.prop("disabled", false);
button_upload.prop("disabled", false);
container_upload.hide();
container_error.hide();
container_error.addClass("hidden");
container_error.addClass("hidden");
modal.set_closeable(true);
};
@ -616,7 +624,8 @@ namespace Modals {
"Succeeded icons: " + succeed_count + "<br>" +
"Failed icons: " + failed_count
);
container_success.css({opacity: 0}).show().animate({opacity: 1}, 250, () => container_success.css({opacity: undefined}));
container_success.removeClass("hidden");
};
button_upload.on('click', event => {
@ -631,9 +640,9 @@ namespace Modals {
button_upload_abort.on('click', event => finish_upload());
container_success.hide();
container_error.addClass("hidden");
container_success.addClass("hidden");
container_upload.hide();
container_error.hide();
}
modal.open();

View file

@ -3,53 +3,215 @@
/// <reference path="../../proto.ts" />
namespace Modals {
type URLGeneratorSettings = {
flag_direct: boolean,
flag_resolved: boolean
}
const DefaultGeneratorSettings: URLGeneratorSettings = {
flag_direct: true,
flag_resolved: false
};
type URLGenerator = {
generate: (properties: {
address: ServerAddress,
resolved_address: ServerAddress
} & URLGeneratorSettings) => string;
setting_available: (key: keyof URLGeneratorSettings) => boolean;
};
const build_url = (base, params) => {
if(Object.keys(params).length == 0)
return base;
return base + "?" + Object.keys(params)
.map(e => e + "=" + encodeURIComponent(params[e]))
.join("&");
};
//TODO: Server password
const url_generators: {[key: string]:URLGenerator} = {
"tea-web": {
generate: properties => {
const address = properties.resolved_address ? properties.resolved_address : properties.address;
const address_str = address.host + (address.port === 9987 ? "" : address.port);
const parameter = "connect_default=" + (properties.flag_direct ? 1 : 0) + "&connect_address=" + encodeURIComponent(address_str);
let pathbase = "";
if(document.location.protocol !== 'https:') {
/*
* Seems to be a test environment or the TeaClient for localhost where we dont have to use https.
*/
pathbase = "https://web.teaspeak.de/";
} else if(document.location.hostname === "localhost" || document.location.host.startsWith("127.")) {
pathbase = "https://web.teaspeak.de/";
} else {
pathbase = document.location.origin + document.location.pathname;
}
return pathbase + "?" + parameter;
},
setting_available: setting => {
return {
flag_direct: true,
flag_resolved: true
}[setting] || false;
}
},
"tea-client": {
generate: properties => {
const address = properties.resolved_address ? properties.resolved_address : properties.address;
let parameters = {
connect_default: properties.flag_direct ? 1 : 0
};
if(address.port != 9987)
parameters["port"] = address.port;
return build_url("teaclient://" + address.host + "/", parameters);
},
setting_available: setting => {
return {
flag_direct: true,
flag_resolved: true
}[setting] || false;
}
},
"teamspeak": {
generate: properties => {
const address = properties.resolved_address ? properties.resolved_address : properties.address;
let parameters = {};
if(address.port != 9987)
parameters["port"] = address.port;
/*
ts3server://<host>?
port=9987
nickname=UserNickname
password=serverPassword
channel=MyDefaultChannel
cid=channelID
channelpassword=defaultChannelPassword
token=TokenKey
addbookmark=MyBookMarkLabel
*/
return build_url("ts3server://" + address.host + "/", parameters);
},
setting_available: setting => {
return {
flag_direct: false,
flag_resolved: true
}[setting] || false;
}
}
};
export function spawnInviteEditor(connection: ConnectionHandler) {
let modal: Modal;
modal = createModal({
header: tr("Invalid URL creator"),
header: tr("Invite URL creator"),
body: () => {
let template = $("#tmpl_invite").renderTag();
template.find(".button-close").on('click', event => modal.close());
return template;
},
footer: undefined
footer: undefined,
min_width: "20em",
width: "50em"
});
const container_url = modal.htmlTag.find(".text-output");
modal.htmlTag.find(".modal-body").addClass("modal-invite");
const button_copy = modal.htmlTag.find(".button-copy");
const input_type = modal.htmlTag.find(".property-type select");
const label_output = modal.htmlTag.find(".text-output");
const invite_settings = [
{
key: "flag_direct",
node: modal.htmlTag.find(".flag-direct-connect input"),
value: node => node.prop('checked'),
set_value: (node, value) => node.prop('checked', value == "1"),
disable: (node, flag) => node.prop('disabled', flag)
.firstParent('.checkbox').toggleClass('disabled', flag)
},
{
key: "flag_resolved",
node: modal.htmlTag.find(".flag-resolved-address input"),
value: node => node.prop('checked'),
set_value: (node, value) => node.prop('checked', value == "1"),
disable: (node, flag) => node.prop('disabled', flag)
.firstParent('.checkbox').toggleClass('disabled', flag)
}
];
const update_buttons = () => {
const generator = url_generators[input_type.val() as string];
if(!generator) {
for(const s of invite_settings)
s.disable(s.node, true);
return;
}
for(const s of invite_settings)
s.disable(s.node, !generator.setting_available(s.key as any));
};
const update_link = () => {
const generator = url_generators[input_type.val() as string];
if(!generator) {
button_copy.prop('disabled', true);
label_output.text(tr("Missing link generator"));
return;
}
button_copy.prop('disabled', false);
const properties = {
address: connection.channelTree.server.remote_address,
resolved_address: connection.channelTree.client.serverConnection.remote_address()
};
for(const s of invite_settings)
properties[s.key] = s.value(s.node);
label_output.text(generator.generate(properties as any));
};
for(const s of invite_settings) {
s.node.on('change keyup', () => {
settings.changeGlobal(Settings.FN_INVITE_LINK_SETTING(s.key), s.value(s.node));
update_link()
});
s.set_value(s.node, settings.global(Settings.FN_INVITE_LINK_SETTING(s.key), DefaultGeneratorSettings[s.key]));
}
input_type.on('change', () => {
settings.changeGlobal(Settings.KEY_LAST_INVITE_LINK_TYPE, input_type.val());
update_buttons();
update_link();
}).val(settings.global(Settings.KEY_LAST_INVITE_LINK_TYPE));
button_copy.on('click', event => {
container_url.select();
label_output.select();
document.execCommand('copy');
});
let flag_direct_connect = true;
let flag_resolved_address = false;
const update_link = () => {
const address = flag_resolved_address ? this.channelTree.client.serverConnection.remote_address() : connection.channelTree.server.remote_address;
const parameter = "connect_default=" + (flag_direct_connect ? 1 : 0) + "&connect_address=" + encodeURIComponent(address.host + (address.port === 9987 ? "" : address.port));
const url = document.location.origin + document.location.pathname + "?" + parameter;
container_url.text(url);
};
{
const input_direct_connect = modal.htmlTag.find(".flag-direct-connect input") as JQuery<HTMLInputElement>;
input_direct_connect.on('change', event => {
flag_direct_connect = input_direct_connect[0].checked;
update_link();
});
input_direct_connect[0].checked = flag_direct_connect;
}
{
const input = modal.htmlTag.find(".flag-resolved-address input") as JQuery<HTMLInputElement>;
input.on('change', event => {
flag_resolved_address = input[0].checked;
update_link();
});
input[0].checked = flag_resolved_address;
}
update_buttons();
update_link();
modal.open();
}
}
}
/*
<option value="tea-web">{{tr "TeaWeb" /}}</option>
<option value="tea-client">{{tr "TeaClient" /}}</option>
<option value="teamspeak">{{tr "TeamSpeak" /}}</option>
*/

View file

@ -69,6 +69,11 @@ namespace Modals {
}
export function spawnPlaylistEdit(client: ConnectionHandler, playlist: Playlist) {
{
createErrorModal(tr("Not implemented"), tr("Playlist editing hasn't yet been implemented")).open();
return;
}
let modal: Modal;
let changed_properties = {};
let changed_permissions = {};

View file

@ -1,9 +1,16 @@
/// <reference path="../../ui/elements/modal.ts" />
/// <reference path="../../i18n/localize.ts" />
/// <reference path="../../ConnectionHandler.ts" />
/// <reference path="../../proto.ts" />
namespace Modals {
export function spawnPlaylistManage(client: ConnectionHandler) {
{
createErrorModal(tr("Not implemented"), tr("Playlist management hasn't yet been implemented")).open();
return;
}
let modal: Modal;
let selected_playlist: Playlist;
let available_playlists: Playlist[];

View file

@ -19,34 +19,31 @@ namespace Modals {
return;
}
//client_login_password
const single_handler: connection.SingleCommandHandler = {
function: command => {
const json = command.arguments[0];
function: command => {
const json = command.arguments[0];
spawnQueryCreated({
username: name,
password: json.client_login_password
}, true);
spawnQueryCreated({
username: name,
password: json.client_login_password
}, true);
if(callback_created)
callback_created(name, json.client_login_password);
return true;
}
if(callback_created)
callback_created(name, json.client_login_password);
return true;
},
command: "notifyquerycreated"
};
connection.serverConnection.command_handler_boss().register_single_handler(single_handler);
connection.serverConnection.send_command("querycreate", {
client_login_name: name
}).catch(error => {
connection.serverConnection.command_handler_boss().remove_single_handler(single_handler);
if(error instanceof CommandResult)
error = error.extra_message || error.message;
createErrorModal(tr("Unable to create account"), tr("Failed to create account<br>Message: ") + error).open();
});
}).then(() => connection.serverConnection.command_handler_boss().remove_single_handler(single_handler));
modal.close();
//TODO create account
});
return template;
},

View file

@ -3,6 +3,7 @@
/// <reference path="../../proto.ts" />
namespace Modals {
/*
export function spawnQueryManage(client: ConnectionHandler) {
let modal: Modal;
let selected_query: QueryListEntry;
@ -57,7 +58,7 @@ namespace Modals {
let template = $("#tmpl_query_manager").renderTag();
template = $.spawn("div").append(template);
/* first open the modal */
/* first open the modal
setTimeout(() => {
const entry_container = template.find(".query-list-entries-container");
if(entry_container.hasScrollBar())
@ -158,4 +159,258 @@ namespace Modals {
update_list();
modal.open();
}
*/
//tmpl_query_manager
export function spawnQueryManage(client: ConnectionHandler) {
let modal: Modal;
modal = createModal({
header: tr("Manage query accounts"),
body: () => {
let template = $("#tmpl_query_manager").renderTag();
let current_server: number;
let selected_query: QueryListEntry;
let filter_callbacks: ((text: string) => boolean)[] = [];
const container_list = template.find(".container-list .container-entries");
const container_list_empty = container_list.find(".container-empty");
const container_list_error = container_list.find(".container-error");
const detail_name = template.find(".detail.login-name .value");
const detail_unique_id = template.find(".detail.unique-id .value");
const detail_bound_server = template.find(".detail.bound-server .value");
const detail_unique_id_copy = template.find(".detail.unique-id .button-copy");
const input_filter = template.find(".filter-input");
const button_create = template.find(".button-create");
const button_delete = template.find(".button-delete");
const button_rename = template.find(".button-rename");
const button_change_password = template.find(".button-change-password");
const button_update = template.find(".button-update");
const permission_create = client.permissions.neededPermission(PermissionType.B_CLIENT_QUERY_CREATE).granted(1);
const permission_delete = client.permissions.neededPermission(PermissionType.B_CLIENT_QUERY_DELETE).granted(1);
const permission_delete_own = client.permissions.neededPermission(PermissionType.B_CLIENT_QUERY_DELETE_OWN).granted(1);
const permission_rename = client.permissions.neededPermission(PermissionType.B_CLIENT_QUERY_RENAME).granted(1);
const permission_rename_own = client.permissions.neededPermission(PermissionType.B_CLIENT_QUERY_RENAME_OWN).granted(1);
const permission_password = client.permissions.neededPermission(PermissionType.B_CLIENT_QUERY_CHANGE_PASSWORD).granted(1);
const permission_password_own = client.permissions.neededPermission(PermissionType.B_CLIENT_QUERY_CHANGE_OWN_PASSWORD).granted(1);
const permission_password_global = client.permissions.neededPermission(PermissionType.B_CLIENT_QUERY_CHANGE_PASSWORD_GLOBAL).granted(1);
button_create.prop('disabled', !permission_create);
const set_error = (error: string | undefined) => {
if(typeof(error) === "string")
container_list_error.text(error).show();
else
container_list_error.hide();
};
const update_list = (selected_entry: string | undefined) => {
button_update.prop('disabled', true);
container_list_empty.text(tr("loading...")).show();
set_error(undefined);
set_selected(undefined, false);
filter_callbacks = [];
container_list.find(".entry").remove();
client.serverConnection.command_helper.current_virtual_server_id().then(server_id => {
current_server = server_id;
client.serverConnection.command_helper.request_query_list(server_id).then(result => {
if(!result || !result.queries.length) {
container_list_empty.text(tr("No queries available"));
return;
}
for(const entry of result.queries) {
const tag = $.spawn("div").addClass("entry").text(entry.username + " (" + entry.unique_id + ")");
tag.on('click', event => {
container_list.find(".selected").removeClass("selected");
tag.addClass("selected");
set_selected(entry, false);
});
container_list.append(tag);
if(entry.username === selected_entry) tag.trigger('click');
const text_mesh = (entry.username + " " + entry.unique_id + " " + entry.bounded_server).toLowerCase();
filter_callbacks.push(text => {
if(typeof(text) === "undefined" || text_mesh.indexOf(text) != -1) {
tag.show();
return true;
} else {
tag.hide();
return false;
}
});
}
update_filter();
container_list_empty.hide();
button_update.prop('disabled', false);
}).catch(error => {
button_update.prop('disabled', false);
if(error instanceof CommandResult && error.id === ErrorID.PERMISSION_ERROR) {
set_error(tr("No permissions"));
return;
}
log.error(LogCategory.CLIENT, tr("Failed to request the query list: %o"), error);
set_error(tr("Failed to request list"));
});
}).catch(error => {
button_update.prop('disabled', false);
log.error(LogCategory.CLIENT, tr("Failed to get own virtual server id: %o"), error);
set_error(tr("Failed to query server id"));
});
};
const set_selected = (entry: QueryListEntry | undefined, force: boolean) => {
if(entry === selected_query && !force) return;
selected_query = entry;
if(!selected_query) {
detail_name.text("-");
detail_unique_id.text("-");
detail_bound_server.text("-");
button_delete.prop('disabled', true);
button_rename.prop('disabled', true);
button_change_password.prop('disabled', true);
} else {
detail_name.text(selected_query.username);
detail_unique_id.text(selected_query.unique_id);
if(selected_query.bounded_server == 0)
detail_bound_server.text(tr("On the instance"));
else if(selected_query.bounded_server === current_server)
detail_bound_server.text(tr("On the current server"));
else
detail_bound_server.text(selected_query.bounded_server.toString());
button_delete.prop('disabled', !permission_delete && !(selected_query.unique_id === client.getClient().properties.client_unique_identifier && permission_delete_own));
button_rename.prop('disabled', !permission_rename && !(selected_query.unique_id === client.getClient().properties.client_unique_identifier && permission_rename_own));
if(selected_query.bounded_server != 0) {
button_change_password.prop('disabled', !permission_password && !(selected_query.unique_id === client.getClient().properties.client_unique_identifier && permission_password_own));
} else {
button_change_password.prop('disabled', !permission_password_global && !(selected_query.unique_id === client.getClient().properties.client_unique_identifier && permission_password_own));
}
}
};
const update_filter = () => {
let value = input_filter.val() as string;
if(!value) value = undefined;
else value = value.toLowerCase();
const shown = filter_callbacks.filter(e => e(value)).length;
if(shown > 0) {
container_list_empty.hide();
} else {
container_list_empty.text(tr("No accounts found")).show();
}
};
input_filter.on('change keyup', update_filter);
/* all buttons */
{
detail_unique_id_copy.on('click', event => {
if(!selected_query) return;
copy_to_clipboard(selected_query.unique_id);
createInfoModal(tr("Unique ID copied"), tr("The unique id has been successfully copied to your clipboard.")).open();
});
button_create.on('click', event => {
Modals.spawnQueryCreate(client, (user, pass) => update_list(user));
});
button_delete.on('click', event => {
if(!selected_query) return;
Modals.spawnYesNo(tr("Are you sure?"), tr("Do you really want to delete this account?"), result => {
if(result) {
client.serverConnection.send_command("querydelete", {
client_login_name: selected_query.username
}).then(() => {
createInfoModal(tr("Account successfully deleted"), tr("The query account has been successfully deleted!")).open();
update_list(undefined);
}).catch(error => {
if(error instanceof CommandResult)
error = error.extra_message || error.message;
createErrorModal(tr("Unable to delete account"), MessageHelper.formatMessage(tr("Failed to delete account{:br:}Message: {}"), error)).open();
});
}
});
});
button_rename.on('click', () => {
if(!selected_query) return;
createInputModal(tr("Change account name"), tr("Enter the new name for the login:"), text => text.length >= 3, result => {
if(result) {
client.serverConnection.send_command("queryrename", {
client_login_name: selected_query.username,
client_new_login_name: result
}).then(() => {
createInfoModal(tr("Account successfully renamed"), tr("The query account has been renamed!")).open();
update_list(result as string);
}).catch(error => {
if(error instanceof CommandResult)
error = error.extra_message || error.message;
createErrorModal(tr("Unable to rename account"), MessageHelper.formatMessage(tr("Failed to rename account{:br:}Message: {}"), error)).open();
});
}
}).open();
});
button_change_password.on('click', () => {
if(!selected_query) return;
createInputModal(tr("Change account's password"), tr("Enter a new password (leave blank for auto generation):"), text => true, result => {
if(result !== false) {
const single_handler: connection.SingleCommandHandler = {
command: "notifyquerypasswordchanges",
function: command => {
Modals.spawnQueryCreated({
username: command.arguments[0]["client_login_name"],
password: command.arguments[0]["client_login_password"]
}, false);
return true;
}
};
client.serverConnection.command_handler_boss().register_single_handler(single_handler);
client.serverConnection.send_command("querychangepassword", {
client_login_name: selected_query.username,
client_login_password: result
}).catch(error => {
client.serverConnection.command_handler_boss().remove_single_handler(single_handler);
if(error instanceof CommandResult)
error = error.extra_message || error.message;
createErrorModal(tr("Unable to change password"), MessageHelper.formatMessage(tr("Failed to change password{:br:}Message: {}"), error)).open();
});
}
}).open();
});
button_update.on('click', event => update_list(selected_query ? selected_query.username : undefined));
}
modal.close_listener.push(() => filter_callbacks = undefined);
set_selected(undefined, true);
update_list(undefined);
template.dividerfy();
return template;
},
footer: null,
min_width: "25em"
});
modal.htmlTag.find(".modal-body").addClass("modal-query-manage");
modal.open();
}
}

View file

@ -260,6 +260,7 @@ class ServerEntry {
type: contextmenu.MenuEntryType.ENTRY,
icon_class: 'client-iconsview',
name: tr("View avatars"),
visible: false, //TODO: Enable again as soon the new design is finished
callback: () => Modals.spawnAvatarList(this.channelTree.client)
},
contextmenu.Entry.CLOSE(() => trigger_close ? on_close() : {})

View file

@ -52,7 +52,7 @@ const loader_javascript = {
} else {
/* test if js/proto.js is available. If so we're in debug mode */
const request = new XMLHttpRequest();
request.open('GET', 'js/proto.js', true);
request.open('GET', "js/proto.js?_ts=" + Date.now(), true);
await new Promise((resolve, reject) => {
request.onreadystatechange = () => {
@ -73,18 +73,6 @@ const loader_javascript = {
}
},
load_scripts: async () => {
/*
if(window.require !== undefined) {
console.log("Loading node specific things");
const remote = require('electron').remote;
module.paths.push(remote.app.getAppPath() + "/node_modules");
module.paths.push(remote.app.getAppPath() + "/app");
module.paths.push(remote.getGlobal("browser-root") + "js/");
window.$ = require("assets/jquery.min.js");
require("native/loader_adapter.js");
}
*/
if(!window.require) {
await loader.load_script(["vendor/jquery/jquery.min.js"]);
} else {
@ -100,20 +88,6 @@ const loader_javascript = {
}
await loader.load_script(["vendor/DOMPurify/purify.min.js"]);
/* bootstrap material design and libs */
//await loader.load_script(["vendor/popper/popper.js"]);
//depends on popper
//await loader.load_script(["vendor/bootstrap-material/bootstrap-material-design.js"]);
/*
loader.register_task(loader.Stage.JAVASCRIPT_INITIALIZING, {
name: "materialize body",
priority: 10,
function: async () => { $(document).ready(function() { $('body').bootstrapMaterialDesign(); }); }
});
*/
await loader.load_script("vendor/jsrender/jsrender.min.js");
await loader.load_scripts([
["vendor/xbbcode/src/parser.js"],
@ -368,6 +342,7 @@ const loader_style = {
"css/static/modal-connect.css",
"css/static/modal-channel.css",
"css/static/modal-query.css",
"css/static/modal-volume.css",
"css/static/modal-invite.css",
"css/static/modal-playlist.css",
"css/static/modal-banlist.css",

View file

@ -1,26 +1,7 @@
- Modals
- Settings (X)
- Query
- List
- Username
- Unique ID
- Bounded server
Buttons:
- Create
- Delete
- Rename
- Change password
- Refresh
- "Ban Client" dialog
- Entity info (Popup)
- Server (Bandwidth (MH))
- Channel
- Icon Select
- Icon upload
- Avatar list
- Invite buddy
- Change volume
- Music System
- Ban Liste
- Übersicht
@ -136,4 +117,5 @@ Fix these icons: https://img.did.science/Screenshot_20-11-06.png
- Application Options
- Crash
- Focus crash window on crash
- Add a notification (Like the browser notifications)
- Add a notification (Like the browser notifications)
Connection state sometimes does not update

View file

@ -4,9 +4,6 @@ html, body {
height: 100%;
width: 100%;
position: fixed;
//min-height: 250px;
//min-width: 250px;
}
.app-container {

View file

@ -0,0 +1 @@
/home/wolverindev/TeaSpeak/Web-Client/web/js/connection/ServerConnection.js

1
web/environment/release/js/client.min.js vendored Symbolic link
View file

@ -0,0 +1 @@
/home/wolverindev/TeaSpeak/Web-Client/web/generated/client.min.js

View file

@ -232,7 +232,7 @@ namespace connection {
this._connected = false;
if(this._voice_connection)
this._voice_connection.dropSession();
this._voice_connection.drop_rtp_session();
}
private handle_socket_message(data) {
@ -264,7 +264,7 @@ namespace connection {
this.do_ping();
this.updateConnectionState(ConnectionState.CONNECTED);
if(this._voice_connection)
this._voice_connection.createSession(); /* FIXME: Move it to a handler boss and not here! */
this._voice_connection.start_rtc_session(); /* FIXME: Move it to a handler boss and not here! */
}
group.end();
} else if(json["type"] === "WebRTC") {

View file

@ -122,6 +122,12 @@ namespace audio {
dataChannel: RTCDataChannel;
private _type: VoiceEncodeType = VoiceEncodeType.NATIVE_ENCODE;
/*
* To ensure we're not sending any audio because the settings activates the input,
* we self mute the audio stream
*/
local_audio_mute: GainNode;
local_audio_stream: MediaStreamAudioDestinationNode;
static codec_pool: codec.CodecPool[];
@ -149,7 +155,7 @@ namespace audio {
destroy() {
clearInterval(this.send_task);
this.dropSession();
this.drop_rtp_session();
this.acquire_voice_recorder(undefined, true).catch(error => {
log.warn(LogCategory.VOICE, tr("Failed to release voice recorder: %o"), error);
}).then(() => {
@ -200,8 +206,14 @@ namespace audio {
return;
}
if(!this.local_audio_stream)
if(!this.local_audio_stream) {
this.local_audio_stream = audio.player.context().createMediaStreamDestination();
}
if(!this.local_audio_mute) {
this.local_audio_mute = audio.player.context().createGain();
this.local_audio_mute.connect(this.local_audio_stream);
this.local_audio_mute.gain.value = 1;
}
}
private setup_js() {
@ -235,16 +247,16 @@ namespace audio {
await recorder.input.set_consumer({
type: audio.recorder.InputConsumerType.NODE,
callback_node: node => {
if(!this.local_audio_stream)
if(!this.local_audio_stream || !this.local_audio_mute)
return;
node.connect(this.local_audio_stream);
node.connect(this.local_audio_mute);
},
callback_disconnect: node => {
if(!this.local_audio_stream)
if(!this.local_audio_mute)
return;
node.disconnect(this.local_audio_stream);
node.disconnect(this.local_audio_mute);
}
} as audio.recorder.NodeInputConsumer);
} else {
@ -266,7 +278,7 @@ namespace audio {
this.setup_native();
else
this.setup_js();
this.createSession();
this.start_rtc_session();
}
voice_playback_support() : boolean {
@ -315,11 +327,14 @@ namespace audio {
}
}
createSession() {
private _audio_player_waiting = false;
start_rtc_session() {
if(!audio.player.initialized()) {
log.info(LogCategory.VOICE, tr("Audio player isn't initialized yet. Waiting for gesture."));
audio.player.on_ready(() => this.createSession());
if(!this._audio_player_waiting) {
this._audio_player_waiting = true;
audio.player.on_ready(() => this.start_rtc_session());
}
return;
}
@ -331,7 +346,7 @@ namespace audio {
else
this.setup_js();
this.dropSession();
this.drop_rtp_session();
this._ice_use_cache = true;
@ -354,17 +369,17 @@ namespace audio {
this.rtcPeerConnection.onicecandidate = this.on_local_ice_candidate.bind(this);
if(this.local_audio_stream) { //May a typecheck?
this.rtcPeerConnection.addStream(this.local_audio_stream.stream);
log.info(LogCategory.VOICE, tr("Adding stream (%o)!"), this.local_audio_stream.stream);
log.info(LogCategory.VOICE, tr("Adding native audio stream (%o)!"), this.local_audio_stream.stream);
}
this.rtcPeerConnection.createOffer(sdpConstraints).then(offer => {
this.on_local_offer_created(offer);
}).catch(error => {
log.error(LogCategory.VOICE, tr("Could not create ice offer! error: %o"), error);
});
this.rtcPeerConnection.createOffer(sdpConstraints)
.then(offer => this.on_local_offer_created(offer))
.catch(error => {
log.error(LogCategory.VOICE, tr("Could not create ice offer! error: %o"), error);
});
}
dropSession() {
drop_rtp_session() {
if(this.dataChannel) {
this.dataChannel.close();
this.dataChannel = undefined;
@ -418,14 +433,13 @@ namespace audio {
});
log.error(LogCategory.NETWORKING, tr("Failed to setup voice bridge (%s). Allow reconnect: %s"), json["reason"], json["allow_reconnect"]);
if(json["allow_reconnect"] == true) {
this.createSession();
this.start_rtc_session();
}
//TODO handle fail specially when its not allowed to reconnect
}
}
}
//Listeners
private on_local_ice_candidate(event: RTCPeerConnectionIceEvent) {
if (event) {
//if(event.candidate && event.candidate.protocol !== "udp")
@ -509,6 +523,7 @@ namespace audio {
const chandler = this.connection.client;
if(!chandler.connected)
return false;
if(chandler.client_status.input_muted)
return false;
@ -544,6 +559,15 @@ namespace audio {
private handle_local_voice_started() {
const chandler = this.connection.client;
if(chandler.client_status.input_muted) {
/* evail hack due to the settings :D */
log.warn(LogCategory.VOICE, tr("Received local voice started event, even thou we're muted! Do not send any voice."));
if(this.local_audio_mute)
this.local_audio_mute.gain.value = 0;
return;
}
if(this.local_audio_mute)
this.local_audio_mute.gain.value = 1;
log.info(LogCategory.VOICE, tr("Local voice started"));
const ch = chandler.getClient();
@ -577,7 +601,7 @@ namespace audio {
unregister_client(client: connection.voice.VoiceClient): Promise<void> {
if(!(client instanceof audio.js.VoiceClientController))
throw "Invalid client";
throw "Invalid client type";
this._audio_clients.remove(client);
return Promise.resolve();
@ -634,12 +658,6 @@ loader.register_task(loader.Stage.JAVASCRIPT_INITIALIZING, {
new audio.js.codec.CodecPool(5, tr("Opus Music"), CodecType.OPUS_MUSIC)
];
if(native_client) {
audio.js.VoiceConnection.codec_pool[0].initialize(2);
audio.js.VoiceConnection.codec_pool[1].initialize(2);
audio.js.VoiceConnection.codec_pool[2].initialize(2);
audio.js.VoiceConnection.codec_pool[3].initialize(2);
}
audio.js.VoiceConnection.codec_pool[4].initialize(2);
audio.js.VoiceConnection.codec_pool[5].initialize(2);
});