add dashbord and right colomn right page
This commit is contained in:
parent
ff3011de69
commit
9a7d23d24d
@ -12,13 +12,13 @@
|
|||||||
import os
|
import os
|
||||||
import logging
|
import logging
|
||||||
import parsecfg
|
import parsecfg
|
||||||
import db_sqlite
|
import db_sqlite
|
||||||
|
|
||||||
import ../code/password_utils
|
import ../code/password_utils
|
||||||
|
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# PROC : Generate Database
|
# PROC : Generate Database
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
proc generateDB*() =
|
proc generateDB*() =
|
||||||
echo "Generating database"
|
echo "Generating database"
|
||||||
@ -26,11 +26,11 @@ proc generateDB*() =
|
|||||||
# Load the connection details
|
# Load the connection details
|
||||||
let
|
let
|
||||||
dict = loadConfig("config/config.cfg")
|
dict = loadConfig("config/config.cfg")
|
||||||
db_user = dict.getSectionValue("Database","user")
|
db_user = dict.getSectionValue("Database", "user")
|
||||||
db_pass = dict.getSectionValue("Database","pass")
|
db_pass = dict.getSectionValue("Database", "pass")
|
||||||
db_name = dict.getSectionValue("Database","name")
|
db_name = dict.getSectionValue("Database", "name")
|
||||||
db_host = dict.getSectionValue("Database","host")
|
db_host = dict.getSectionValue("Database", "host")
|
||||||
db_folder = dict.getSectionValue("Database","folder")
|
db_folder = dict.getSectionValue("Database", "folder")
|
||||||
dbexists = if fileExists(db_host): true else: false
|
dbexists = if fileExists(db_host): true else: false
|
||||||
|
|
||||||
if dbexists:
|
if dbexists:
|
||||||
@ -41,7 +41,8 @@ proc generateDB*() =
|
|||||||
|
|
||||||
# Open DB
|
# Open DB
|
||||||
echo " - Opening database"
|
echo " - Opening database"
|
||||||
var db = open(connection=db_host, user=db_user, password=db_pass, database=db_name)
|
var db = open(connection = db_host, user = db_user, password = db_pass,
|
||||||
|
database = db_name)
|
||||||
|
|
||||||
# Person table contains information about the
|
# Person table contains information about the
|
||||||
# registrered users
|
# registrered users
|
||||||
@ -74,6 +75,20 @@ proc generateDB*() =
|
|||||||
);""")):
|
);""")):
|
||||||
echo " - Database: session table already exists"
|
echo " - Database: session table already exists"
|
||||||
|
|
||||||
|
# Sesion table contains information about the search
|
||||||
|
#
|
||||||
|
if not db.tryExec(sql("""
|
||||||
|
create table if not exists search(
|
||||||
|
id integer primary key,
|
||||||
|
search_title varchar(300) not null,
|
||||||
|
query varchar(300) not null,
|
||||||
|
search_type varchar(100) not null,
|
||||||
|
creation timestamp not null default (STRFTIME('%s', 'now')),
|
||||||
|
lastModified timestamp not null default (STRFTIME('%s', 'now')),
|
||||||
|
foreign key (userid) references person(id)
|
||||||
|
);""")):
|
||||||
|
echo " - Database: search table already exists"
|
||||||
|
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# PROC : Create Administrator User
|
# PROC : Create Administrator User
|
||||||
@ -108,9 +123,10 @@ proc createAdminUser*(db: DbConn, args: seq[string]) =
|
|||||||
let password = makePassword(iPwd, salt)
|
let password = makePassword(iPwd, salt)
|
||||||
|
|
||||||
# Insert user into database
|
# Insert user into database
|
||||||
if insertID(db, sql"INSERT INTO person (name, email, password, salt, status) VALUES (?, ?, ?, ?, ?)", $iName, $iEmail, password, salt, "Admin") > 0:
|
if insertID(db, sql"INSERT INTO person (name, email, password, salt, status) VALUES (?, ?, ?, ?, ?)",
|
||||||
|
$iName, $iEmail, password, salt, "Admin") > 0:
|
||||||
echo "Admin user added"
|
echo "Admin user added"
|
||||||
else:
|
else:
|
||||||
error("Something went wrong")
|
error("Something went wrong")
|
||||||
|
|
||||||
info("Admin added.")
|
info("Admin added.")
|
||||||
|
@ -15,10 +15,13 @@ import std/json
|
|||||||
import httpcore
|
import httpcore
|
||||||
import strutils
|
import strutils
|
||||||
import strformat
|
import strformat
|
||||||
|
import std/times
|
||||||
import std/options
|
import std/options
|
||||||
import std/httpclient
|
import std/httpclient
|
||||||
import std/asyncdispatch
|
import std/asyncdispatch
|
||||||
|
|
||||||
|
import utils # Joplin utils procedures and types
|
||||||
|
|
||||||
from os import sleep
|
from os import sleep
|
||||||
from posix import read, write, fdatasync, close
|
from posix import read, write, fdatasync, close
|
||||||
|
|
||||||
@ -37,16 +40,18 @@ type
|
|||||||
type
|
type
|
||||||
joplin_tags* = object
|
joplin_tags* = object
|
||||||
id*, parent_id*, title*: seq[string]
|
id*, parent_id*, title*: seq[string]
|
||||||
created_time*, updated_time*, user_created_time*, user_updated_time*, is_shared*: seq[int]
|
created_time*, updated_time*, user_created_time*, user_updated_time*,
|
||||||
|
is_shared*: seq[int]
|
||||||
req*: Request
|
req*: Request
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# TYPE : Setup joplin_notebooks data
|
# TYPE : Setup joplin_notebooks data
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
type
|
type
|
||||||
joplin_notebooks* = object
|
joplin_notebooks* = object
|
||||||
id*, parent_id*, title*, share_id*, icon*: seq[string]
|
id*, parent_id*, title*, share_id*, icon*: seq[string]
|
||||||
created_time*, updated_time*, user_created_time*, user_updated_time*, is_shared*: seq[int]
|
created_time*, updated_time*, user_created_time*, user_updated_time*,
|
||||||
|
is_shared*: seq[int]
|
||||||
req*: Request
|
req*: Request
|
||||||
|
|
||||||
|
|
||||||
@ -54,10 +59,29 @@ type
|
|||||||
# TYPE : Setup joplin_notes data
|
# TYPE : Setup joplin_notes data
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
type
|
type
|
||||||
joplin_notes* = object
|
joplin_notes* = object
|
||||||
id*, parent_id*, title*, body*, author*, source_url*, source*, source_application*, application_data*, share_id*, conflict_original_id*, body_html*, base_url*: seq[string]
|
id*, parent_id*, title*, body*, author*, source_url*, source*,
|
||||||
created_time*, updated_time*, is_conflict*, is_todo*, todo_due*, todo_completed*, user_created_time*, user_updated_time*, markup_language*, is_shared*: seq[int]
|
source_application*, application_data*, share_id*,
|
||||||
latitude*, longitude*, altitude*, order*:seq[float]
|
conflict_original_id*, body_html*, base_url*: seq[string]
|
||||||
|
updated_time*, is_conflict*, is_todo*, todo_due*,
|
||||||
|
todo_completed*, user_created_time*, user_updated_time*,
|
||||||
|
markup_language*, is_shared*: seq[int]
|
||||||
|
latitude*, longitude*, altitude*, order*: seq[float]
|
||||||
|
created_time*: seq[string]
|
||||||
|
req*: Request
|
||||||
|
|
||||||
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
|
# TYPE : Setup joplin_note data
|
||||||
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
|
type
|
||||||
|
joplin_note* = object
|
||||||
|
id*, parent_id*, title*, body*, author*, source_url*, source*,
|
||||||
|
source_application*, application_data*, conflict_original_id*,
|
||||||
|
master_key_id*, share_id*, encryption_cipher_text * : string
|
||||||
|
created_time*, updated_time*, user_created_time*, user_updated_time*,
|
||||||
|
is_conflict*, is_todo*, todo_due*, todo_completed*, markup_language*,
|
||||||
|
is_shared*, encryption_applied*: int
|
||||||
|
latitude*, longitude*, altitude*, order*: float
|
||||||
req*: Request
|
req*: Request
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
@ -73,7 +97,7 @@ type
|
|||||||
# DURATION : duration variables
|
# DURATION : duration variables
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
let
|
let
|
||||||
resetDuration = initDuration(seconds=2)
|
resetDuration = initDuration(seconds = 2)
|
||||||
deciSecondDuration* = initDuration(milliseconds = 100)
|
deciSecondDuration* = initDuration(milliseconds = 100)
|
||||||
qtrsecondDuration* = initDuration(milliseconds = 250)
|
qtrsecondDuration* = initDuration(milliseconds = 250)
|
||||||
|
|
||||||
@ -86,7 +110,7 @@ var
|
|||||||
|
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# PROC : reset HTTP client
|
# PROC : reset HTTP client
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
proc resetHttpClient() =
|
proc resetHttpClient() =
|
||||||
if (getTime().utc - lastConnection) > resetDuration:
|
if (getTime().utc - lastConnection) > resetDuration:
|
||||||
@ -99,13 +123,13 @@ proc resetHttpClient() =
|
|||||||
except:
|
except:
|
||||||
echo("Attempted to close clients. Probably do not exist.")
|
echo("Attempted to close clients. Probably do not exist.")
|
||||||
echo("Current exception: ", getCurrentExceptionMsg())
|
echo("Current exception: ", getCurrentExceptionMsg())
|
||||||
client = newHttpClient(timeout=500)
|
client = newHttpClient(timeout = 500)
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==--==--==--==-==--==--==- #
|
# --==--==--==--==--==--==--==--==--==--==--==--==--==-==--==--==- #
|
||||||
# PROC : use the joplin API ping to validate service availibility
|
# PROC : use the joplin API ping to validate service availibility
|
||||||
# --==--==--==--==--==--==--==--==--==--==--==--==--==-==--==--==- #
|
# --==--==--==--==--==--==--==--==--==--==--==--==--==-==--==--==- #
|
||||||
proc ping_joplin*(token:string): Future[joplin_ping] {.async.} =
|
proc ping_joplin*(token: string): Future[joplin_ping] {.async.} =
|
||||||
|
|
||||||
var j_p: joplin_ping
|
var j_p: joplin_ping
|
||||||
var client = newAsyncHttpClient()
|
var client = newAsyncHttpClient()
|
||||||
var url: string
|
var url: string
|
||||||
@ -114,7 +138,7 @@ proc ping_joplin*(token:string): Future[joplin_ping] {.async.} =
|
|||||||
url = fmt"http://localhost:41184/ping?token={token}"
|
url = fmt"http://localhost:41184/ping?token={token}"
|
||||||
var json = await client.getContent(url)
|
var json = await client.getContent(url)
|
||||||
j_p.ping_result.add(json)
|
j_p.ping_result.add(json)
|
||||||
|
|
||||||
echo j_p.ping_result[0]
|
echo j_p.ping_result[0]
|
||||||
|
|
||||||
if j_p.ping_result[0] == "JoplinClipperServer":
|
if j_p.ping_result[0] == "JoplinClipperServer":
|
||||||
@ -133,17 +157,17 @@ proc ping_joplin*(token:string): Future[joplin_ping] {.async.} =
|
|||||||
echo("Current Exception: ", getCurrentException().name)
|
echo("Current Exception: ", getCurrentException().name)
|
||||||
echo("Current Exception Msg: ", getCurrentExceptionMsg())
|
echo("Current Exception Msg: ", getCurrentExceptionMsg())
|
||||||
echo("Sleeping for 1 seconds at: ", getTime().utc)
|
echo("Sleeping for 1 seconds at: ", getTime().utc)
|
||||||
j_p.ping_status = false
|
j_p.ping_status = false
|
||||||
return j_p
|
return j_p
|
||||||
# parse jason
|
# parse jason
|
||||||
#let joplin_notes_Json = parseJson(json)
|
#let joplin_notes_Json = parseJson(json)
|
||||||
|
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# PROC : get all joplin NOTEBOOKS
|
# PROC : get all joplin NOTEBOOKS
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
proc get_joplin_notebooks*(token:string): Future[joplin_notebooks] {.async.} =
|
proc get_joplin_notebooks*(token: string): Future[joplin_notebooks] {.async.} =
|
||||||
|
|
||||||
# Variables
|
# Variables
|
||||||
var j_nb: joplin_notebooks
|
var j_nb: joplin_notebooks
|
||||||
var has_more: bool = true
|
var has_more: bool = true
|
||||||
@ -151,15 +175,15 @@ proc get_joplin_notebooks*(token:string): Future[joplin_notebooks] {.async.} =
|
|||||||
var url: string
|
var url: string
|
||||||
var client = newAsyncHttpClient()
|
var client = newAsyncHttpClient()
|
||||||
var pingCheck: joplin_ping
|
var pingCheck: joplin_ping
|
||||||
|
|
||||||
pingCheck = waitFor ping_joplin(token)
|
pingCheck = waitFor ping_joplin(token)
|
||||||
|
|
||||||
# check the joplin status befor query
|
# check the joplin status befor query
|
||||||
if pingCheck.ping_status:
|
if pingCheck.ping_status:
|
||||||
|
|
||||||
# make sure to check all pages
|
# make sure to check all pages
|
||||||
while has_more == true:
|
while has_more == true:
|
||||||
|
|
||||||
# request joplin API for notebooks
|
# request joplin API for notebooks
|
||||||
url = fmt"http://localhost:41184/folders?page={page}&token={token}"
|
url = fmt"http://localhost:41184/folders?page={page}&token={token}"
|
||||||
echo("URL notebooks : ", url)
|
echo("URL notebooks : ", url)
|
||||||
@ -168,10 +192,10 @@ proc get_joplin_notebooks*(token:string): Future[joplin_notebooks] {.async.} =
|
|||||||
# parse jason
|
# parse jason
|
||||||
let joplin_notebooks_Json = parseJson(json)
|
let joplin_notebooks_Json = parseJson(json)
|
||||||
|
|
||||||
# valider qu'il n'y a plus de page
|
# valider qu'il n'y a plus de page
|
||||||
if not joplin_notebooks_Json["has_more"].getBool:
|
if not joplin_notebooks_Json["has_more"].getBool:
|
||||||
has_more = false
|
has_more = false
|
||||||
|
|
||||||
# store json info into an object
|
# store json info into an object
|
||||||
var count: int = 1
|
var count: int = 1
|
||||||
for nb in joplin_notebooks_Json["items"]:
|
for nb in joplin_notebooks_Json["items"]:
|
||||||
@ -179,18 +203,18 @@ proc get_joplin_notebooks*(token:string): Future[joplin_notebooks] {.async.} =
|
|||||||
j_nb.parent_id.add(nb["parent_id"].getstr)
|
j_nb.parent_id.add(nb["parent_id"].getstr)
|
||||||
j_nb.title.add(nb["title"].getstr)
|
j_nb.title.add(nb["title"].getstr)
|
||||||
count += 1
|
count += 1
|
||||||
|
|
||||||
# aller à la page suivante
|
# aller à la page suivante
|
||||||
page += 1
|
page += 1
|
||||||
|
|
||||||
return j_nb
|
return j_nb
|
||||||
|
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# PROC : get all joplin NOTES
|
# PROC : get all joplin NOTES
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
proc get_joplin_notes*(token:string): Future[joplin_notes] {.async.} =
|
proc get_joplin_notes*(token: string): Future[joplin_notes] {.async.} =
|
||||||
|
|
||||||
# Variables
|
# Variables
|
||||||
var j_notes: joplin_notes
|
var j_notes: joplin_notes
|
||||||
var has_more: bool = true
|
var has_more: bool = true
|
||||||
@ -200,43 +224,113 @@ proc get_joplin_notes*(token:string): Future[joplin_notes] {.async.} =
|
|||||||
var pingCheck: joplin_ping
|
var pingCheck: joplin_ping
|
||||||
pingCheck = waitFor ping_joplin(token)
|
pingCheck = waitFor ping_joplin(token)
|
||||||
|
|
||||||
# check the joplin status befor query
|
# check the joplin status befor query
|
||||||
if pingCheck.ping_status:
|
if pingCheck.ping_status:
|
||||||
|
|
||||||
# make sure to check all pages
|
# make sure to check all pages
|
||||||
while has_more == true:
|
while has_more == true:
|
||||||
|
|
||||||
# request joplin API for Notes
|
# request joplin API for Notes
|
||||||
url = fmt"http://localhost:41184/notes?page={page}&token={token}"
|
url = fmt"http://localhost:41184/notes?page={page}&fields=id,parent_id,title,created_time,updated_time&token={token}"
|
||||||
echo("URL notes : ", url)
|
echo("URL notes : ", url)
|
||||||
var json = await client.getContent(url)
|
var json = await client.getContent(url)
|
||||||
|
|
||||||
# parse jason
|
# parse jason
|
||||||
let joplin_notes_Json = parseJson(json)
|
let joplin_notes_Json = parseJson(json)
|
||||||
|
|
||||||
# valider qu'il n'y a plus de page
|
# valider qu'il n'y a plus de page
|
||||||
if not joplin_notes_Json["has_more"].getBool:
|
if not joplin_notes_Json["has_more"].getBool:
|
||||||
has_more = false
|
has_more = false
|
||||||
|
|
||||||
|
#echo joplin_notes_Json
|
||||||
|
|
||||||
# store json info into an object
|
# store json info into an object
|
||||||
var count: int = 1
|
var count: int = 1
|
||||||
|
var epochTime: int
|
||||||
|
var humanTime: Time
|
||||||
for nb in joplin_notes_Json["items"]:
|
for nb in joplin_notes_Json["items"]:
|
||||||
j_notes.id.add(nb["id"].getstr)
|
j_notes.id.add(nb["id"].getstr)
|
||||||
j_notes.parent_id.add(nb["parent_id"].getstr)
|
j_notes.parent_id.add(nb["parent_id"].getstr)
|
||||||
j_notes.title.add(nb["title"].getstr)
|
j_notes.title.add(nb["title"].getstr)
|
||||||
|
#epochTime = nb["created_time"].getInt
|
||||||
|
#humanTime = fromUnix(convert(Milliseconds, Seconds, epochTime))
|
||||||
|
|
||||||
|
j_notes.created_time.add(convertEpochToHumanTime(nb[
|
||||||
|
"created_time"].getInt))
|
||||||
count += 1
|
count += 1
|
||||||
|
|
||||||
# aller à la page suivante
|
# aller à la page suivante
|
||||||
page += 1
|
page += 1
|
||||||
|
|
||||||
return j_notes
|
return j_notes
|
||||||
|
|
||||||
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
|
# PROC : get joplin NOTE
|
||||||
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
|
proc get_joplin_note*(token: string, noteid: string): Future[
|
||||||
|
joplin_note] {.async.} =
|
||||||
|
|
||||||
|
# Variables
|
||||||
|
var j_note: joplin_note
|
||||||
|
var has_more: bool = true
|
||||||
|
var page: int = 1
|
||||||
|
var url: string
|
||||||
|
var client = newAsyncHttpClient()
|
||||||
|
|
||||||
|
# Check joplin connection availibility
|
||||||
|
var pingCheck: joplin_ping
|
||||||
|
pingCheck = waitFor ping_joplin(token)
|
||||||
|
|
||||||
|
|
||||||
|
# Query for a selected note
|
||||||
|
url = fmt"http://localhost:41184/notes/{noteid}?fields=id,parent_id,title,body,created_time,updated_time,is_conflict,latitude,longitude,altitude,author,source_url,is_todo,todo_due,todo_completed,source,source_application,application_data,order,user_created_time,user_updated_time,encryption_cipher_text,encryption_applied,markup_language,is_shared,share_id,conflict_original_id,master_key_id&token={token}"
|
||||||
|
echo("URL notes : ", url)
|
||||||
|
var jsonNote = await client.getContent(url)
|
||||||
|
|
||||||
|
# parse jason
|
||||||
|
let joplin_note_Json = parseJson(jsonNote)
|
||||||
|
|
||||||
|
# get all informations for a selected note
|
||||||
|
j_note.id = joplin_note_Json["id"].getstr
|
||||||
|
j_note.parent_id = joplin_note_Json["parent_id"].getstr
|
||||||
|
j_note.title = joplin_note_Json["title"].getstr
|
||||||
|
j_note.body = joplin_note_Json["body"].getstr
|
||||||
|
j_note.author = joplin_note_Json["author"].getstr
|
||||||
|
j_note.source_url = joplin_note_Json["source_url"].getstr
|
||||||
|
j_note.source = joplin_note_Json["source"].getstr
|
||||||
|
j_note.source_application = joplin_note_Json["source_application"].getstr
|
||||||
|
j_note.application_data = joplin_note_Json["application_data"].getstr
|
||||||
|
j_note.share_id = joplin_note_Json["share_id"].getstr
|
||||||
|
j_note.conflict_original_id = joplin_note_Json["conflict_original_id"].getstr
|
||||||
|
j_note.created_time = joplin_note_Json["created_time"].getInt
|
||||||
|
j_note.updated_time = joplin_note_Json["updated_time"].getInt
|
||||||
|
j_note.is_conflict = joplin_note_Json["is_conflict"].getInt
|
||||||
|
j_note.latitude = joplin_note_Json["latitude"].getFloat
|
||||||
|
j_note.longitude = joplin_note_Json["longitude"].getFloat
|
||||||
|
j_note.altitude = joplin_note_Json["altitude"].getFloat
|
||||||
|
j_note.is_todo = joplin_note_Json["is_todo"].getInt
|
||||||
|
j_note.todo_due = joplin_note_Json["todo_due"].getInt
|
||||||
|
j_note.todo_completed = joplin_note_Json["todo_completed"].getInt
|
||||||
|
j_note.order = joplin_note_Json["order"].getFloat
|
||||||
|
j_note.user_created_time = joplin_note_Json["user_created_time"].getInt
|
||||||
|
j_note.user_updated_time = joplin_note_Json["user_updated_time"].getInt
|
||||||
|
j_note.encryption_cipher_text = joplin_note_Json[
|
||||||
|
"encryption_cipher_text"].getstr
|
||||||
|
j_note.encryption_applied = joplin_note_Json["encryption_applied"].getInt
|
||||||
|
j_note.markup_language = joplin_note_Json["markup_language"].getInt
|
||||||
|
j_note.is_shared = joplin_note_Json["is_shared"].getInt
|
||||||
|
j_note.master_key_id = joplin_note_Json["master_key_id"].getstr
|
||||||
|
|
||||||
|
|
||||||
|
echo j_note
|
||||||
|
return j_note
|
||||||
|
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# PROC : get all joplin TAGS
|
# PROC : get all joplin TAGS
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
proc get_joplin_tags*(token:string): Future[joplin_tags] {.async.} =
|
proc get_joplin_tags*(token: string): Future[joplin_tags] {.async.} =
|
||||||
|
|
||||||
# Variables
|
# Variables
|
||||||
var j_tags: joplin_tags
|
var j_tags: joplin_tags
|
||||||
var has_more: bool = true
|
var has_more: bool = true
|
||||||
@ -246,12 +340,12 @@ proc get_joplin_tags*(token:string): Future[joplin_tags] {.async.} =
|
|||||||
var pingCheck: joplin_ping
|
var pingCheck: joplin_ping
|
||||||
pingCheck = waitFor ping_joplin(token)
|
pingCheck = waitFor ping_joplin(token)
|
||||||
|
|
||||||
# check the joplin status befor query
|
# check the joplin status befor query
|
||||||
if pingCheck.ping_status:
|
if pingCheck.ping_status:
|
||||||
|
|
||||||
# make sure to check all pages
|
# make sure to check all pages
|
||||||
while has_more == true:
|
while has_more == true:
|
||||||
|
|
||||||
# request joplin API for notebooks
|
# request joplin API for notebooks
|
||||||
url = fmt"http://localhost:41184/tags?page={page}&token={token}"
|
url = fmt"http://localhost:41184/tags?page={page}&token={token}"
|
||||||
echo("URL tags : ", url)
|
echo("URL tags : ", url)
|
||||||
@ -260,10 +354,10 @@ proc get_joplin_tags*(token:string): Future[joplin_tags] {.async.} =
|
|||||||
# parse jason
|
# parse jason
|
||||||
let joplin_tags_Json = parseJson(json)
|
let joplin_tags_Json = parseJson(json)
|
||||||
|
|
||||||
# valider qu'il n'y a plus de page
|
# valider qu'il n'y a plus de page
|
||||||
if not joplin_tags_Json["has_more"].getBool:
|
if not joplin_tags_Json["has_more"].getBool:
|
||||||
has_more = false
|
has_more = false
|
||||||
|
|
||||||
# store json info into an object
|
# store json info into an object
|
||||||
var count: int = 1
|
var count: int = 1
|
||||||
for nb in joplin_tags_Json["items"]:
|
for nb in joplin_tags_Json["items"]:
|
||||||
@ -271,9 +365,9 @@ proc get_joplin_tags*(token:string): Future[joplin_tags] {.async.} =
|
|||||||
j_tags.parent_id.add(nb["parent_id"].getstr)
|
j_tags.parent_id.add(nb["parent_id"].getstr)
|
||||||
j_tags.title.add(nb["title"].getstr)
|
j_tags.title.add(nb["title"].getstr)
|
||||||
count += 1
|
count += 1
|
||||||
|
|
||||||
# aller à la page suivante
|
# aller à la page suivante
|
||||||
page += 1
|
page += 1
|
||||||
|
|
||||||
return j_tags
|
return j_tags
|
||||||
|
|
||||||
@ -287,42 +381,41 @@ proc get_joplin_cli_token*(): string =
|
|||||||
var result = execCmdEx("joplin config api.token")
|
var result = execCmdEx("joplin config api.token")
|
||||||
|
|
||||||
if result.exitCode == 0:
|
if result.exitCode == 0:
|
||||||
let param1 = result.output
|
let param1 = result.output
|
||||||
let flagSplit = param1.split(" = ")
|
let flagSplit = param1.split(" = ")
|
||||||
flagName = flagSplit[0]
|
flagName = flagSplit[0]
|
||||||
flagValue = flagSplit[1]
|
flagValue = flagSplit[1]
|
||||||
|
|
||||||
return flagValue
|
return flagValue
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# PROC : launch any program and get PID
|
# PROC : launch any program and get PID
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
proc launchProgram(app:string = "", workingPath:string = "", arguments:array[2, string]):int {.thread.} =
|
proc launchProgram(app: string = "", workingPath: string = "", arguments: array[
|
||||||
|
2, string]): int {.thread.} =
|
||||||
var p = startProcess(joinPath(workingPath,app), workingPath, arguments)
|
|
||||||
let pid = p.processID()
|
var p = startProcess(joinPath(workingPath, app), workingPath, arguments)
|
||||||
var outhdl = outputHandle(p)
|
let pid = p.processID()
|
||||||
var inputhdl = inputHandle(p)
|
var outhdl = outputHandle(p)
|
||||||
var errhdl = errorHandle(p)
|
var inputhdl = inputHandle(p)
|
||||||
return pid
|
var errhdl = errorHandle(p)
|
||||||
|
return pid
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# PROC : Start joplin cli
|
# PROC : Start joplin cli
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
proc joplin_cli_start*():int {.thread.} =
|
proc joplin_cli_start*(): int {.thread.} =
|
||||||
|
|
||||||
let joplinProcessId = launchProgram("joplin","/usr/bin/",["server","start"])
|
let joplinProcessId = launchProgram("joplin", "/usr/bin/", ["server", "start"])
|
||||||
return joplinProcessId
|
return joplinProcessId
|
||||||
|
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# PROC : stop Joplin Terminal
|
# PROC : stop Joplin Terminal
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
proc joplin_cli_stop*():int {.thread.} =
|
proc joplin_cli_stop*(): int {.thread.} =
|
||||||
let joplinProcessId = launchProgram("joplin","/usr/bin/",["server","stop"])
|
let joplinProcessId = launchProgram("joplin", "/usr/bin/", ["server", "stop"])
|
||||||
return joplinProcessId
|
return joplinProcessId
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
@ -333,53 +426,45 @@ proc joplin_cli_status*(): bool =
|
|||||||
var result = execCmdEx("joplin server status")
|
var result = execCmdEx("joplin server status")
|
||||||
|
|
||||||
if result.exitCode == 0:
|
if result.exitCode == 0:
|
||||||
if "Server is not running" in result.output :
|
if "Server is not running" in result.output:
|
||||||
echo "Joplin Terminal cli status is down : ", result.output
|
echo "Joplin Terminal cli status is down : ", result.output
|
||||||
rc = false
|
rc = false
|
||||||
else:
|
else:
|
||||||
echo "Joplin Terminal cli status is up : ", result.output
|
echo "Joplin Terminal cli status is up : ", result.output
|
||||||
rc = true
|
rc = true
|
||||||
else:
|
else:
|
||||||
echo "Error validate joplin terminal status : ", result.output
|
echo "Error validate joplin terminal status : ", result.output
|
||||||
return rc
|
return rc
|
||||||
|
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# PROC : start Joplin Terminal
|
# PROC : start or stop Joplin Terminal
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# proc joplin_cli_start*(): bool =
|
proc joplin_cli_start_stop*(): int =
|
||||||
# var rc = false
|
var isStart: int = 0
|
||||||
# startProcess("joplin","/usr/bin", @["server", "start"])
|
var sleep_delay_frame: int = 50
|
||||||
# #var result = execCmdEx("ls -l /usr/bin/joplin")
|
var sleep_max: int = 5000
|
||||||
|
|
||||||
# if joplin_cli_status() == true:
|
if joplin_cli_status() == false:
|
||||||
# echo "Joplin Terminal started successfully : ",result.output
|
isStart = joplin_cli_start()
|
||||||
# rc = true
|
while joplin_cli_status() == false:
|
||||||
# else:
|
sleep(sleep_delay_frame)
|
||||||
# echo "Joplin Terminal didn't start : ",result.output
|
echo "Joplin client Terminal started: ", isStart
|
||||||
# rc = false
|
|
||||||
# return rc
|
else:
|
||||||
|
echo "Joplin client Terminal is alredy started !"
|
||||||
|
#if joplin_cli_status() == true:
|
||||||
|
isStart = joplin_cli_stop()
|
||||||
|
while joplin_cli_status() == true:
|
||||||
|
sleep(sleep_delay_frame)
|
||||||
|
echo "Joplin client Terminal stopped: ", isStart
|
||||||
|
|
||||||
|
return isStart
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# PROC : stop Joplin Terminal
|
# PROC :
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
|
||||||
# proc joplin_cli_stop*(): bool =
|
|
||||||
# var rc = false
|
|
||||||
# var result = execCmdEx("joplin server stop")
|
|
||||||
|
|
||||||
# if result.exitCode == 0:
|
|
||||||
# # echo result.output
|
|
||||||
# rc = true
|
|
||||||
# else:
|
|
||||||
# rc = false
|
|
||||||
# return rc
|
|
||||||
|
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
|
||||||
# PROC :
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
|
|
||||||
# proc get_joplin_tags_json*(token:string): Future[] {.async.} =
|
# proc get_joplin_tags_json*(token:string): Future[] {.async.} =
|
||||||
@ -389,7 +474,7 @@ proc joplin_cli_status*(): bool =
|
|||||||
# # url = fmt"http://localhost:41184/tags/{id}?fields=id,parent_id,title,created_time,updated_time,user_created_time,user_updated_time,is_shared&token={token}"
|
# # url = fmt"http://localhost:41184/tags/{id}?fields=id,parent_id,title,created_time,updated_time,user_created_time,user_updated_time,is_shared&token={token}"
|
||||||
|
|
||||||
# # http://localhost:41184/tags?token=e5f6644fbf6a97ddc55648dae72b11caecda6c6642d8ce0d3b20129b89b196385737eb908923542c3343649ebbf865b55bda031ab4c3a16edc7723ef2ad77d8f
|
# # http://localhost:41184/tags?token=e5f6644fbf6a97ddc55648dae72b11caecda6c6642d8ce0d3b20129b89b196385737eb908923542c3343649ebbf865b55bda031ab4c3a16edc7723ef2ad77d8f
|
||||||
|
|
||||||
# # Variables
|
# # Variables
|
||||||
# var j_tags: joplin_tags
|
# var j_tags: joplin_tags
|
||||||
# var has_more: bool = true
|
# var has_more: bool = true
|
||||||
@ -398,14 +483,14 @@ proc joplin_cli_status*(): bool =
|
|||||||
# var client = newAsyncHttpClient()
|
# var client = newAsyncHttpClient()
|
||||||
# var pingCheck: joplin_ping
|
# var pingCheck: joplin_ping
|
||||||
# pingCheck = waitFor ping_joplin(token)
|
# pingCheck = waitFor ping_joplin(token)
|
||||||
|
|
||||||
|
|
||||||
# # check the joplin status befor query
|
|
||||||
|
# # check the joplin status befor query
|
||||||
# if pingCheck.ping_status:
|
# if pingCheck.ping_status:
|
||||||
|
|
||||||
# # make sure to check all pages
|
# # make sure to check all pages
|
||||||
# while has_more == true:
|
# while has_more == true:
|
||||||
|
|
||||||
# # request joplin API for notebooks
|
# # request joplin API for notebooks
|
||||||
# url = fmt"http://localhost:41184/tags?page={page}&token={token}"
|
# url = fmt"http://localhost:41184/tags?page={page}&token={token}"
|
||||||
# echo("URL tags : ", url)
|
# echo("URL tags : ", url)
|
||||||
@ -414,10 +499,10 @@ proc joplin_cli_status*(): bool =
|
|||||||
# # parse jason
|
# # parse jason
|
||||||
# let joplin_tags_Json = parseJson(json)
|
# let joplin_tags_Json = parseJson(json)
|
||||||
|
|
||||||
# # valider qu'il n'y a plus de page
|
# # valider qu'il n'y a plus de page
|
||||||
# if not joplin_tags_Json["has_more"].getBool:
|
# if not joplin_tags_Json["has_more"].getBool:
|
||||||
# has_more = false
|
# has_more = false
|
||||||
|
|
||||||
# # store json info into an object
|
# # store json info into an object
|
||||||
# var count: int = 1
|
# var count: int = 1
|
||||||
# for nb in joplin_tags_Json["items"]:
|
# for nb in joplin_tags_Json["items"]:
|
||||||
@ -425,8 +510,8 @@ proc joplin_cli_status*(): bool =
|
|||||||
# echo nb["parent_id"].getstr
|
# echo nb["parent_id"].getstr
|
||||||
# echo nb["title"].getstr
|
# echo nb["title"].getstr
|
||||||
# count += 1
|
# count += 1
|
||||||
|
|
||||||
# # aller à la page suivante
|
# # aller à la page suivante
|
||||||
# page += 1
|
# page += 1
|
||||||
|
|
||||||
|
|
||||||
|
@ -25,7 +25,7 @@ let useUrandom = urandom.open("/dev/urandom")
|
|||||||
## : Generate random salt. Uses cryptographically secure /dev/urandom
|
## : Generate random salt. Uses cryptographically secure /dev/urandom
|
||||||
## : on platforms where it is available, and Nim's random module in other cases.
|
## : on platforms where it is available, and Nim's random module in other cases.
|
||||||
# --==--==--==--==--==--==--==--==--==--==--=--==--==--=--==--==--=--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==--=--==--==--=--==--==--=--==--==-- #
|
||||||
proc makeSalt*(): string =
|
proc makeSalt*(): string =
|
||||||
result = ""
|
result = ""
|
||||||
if useUrandom:
|
if useUrandom:
|
||||||
var randomBytes: array[0..127, char]
|
var randomBytes: array[0..127, char]
|
||||||
@ -42,7 +42,7 @@ proc makeSalt*(): string =
|
|||||||
# PROC : Make session key
|
# PROC : Make session key
|
||||||
## Creates a random key to be used to authorize a session.
|
## Creates a random key to be used to authorize a session.
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
proc makeSessionKey*(): string =
|
proc makeSessionKey*(): string =
|
||||||
let random = makeSalt()
|
let random = makeSalt()
|
||||||
return bcrypt.hash(random, genSalt(8))
|
return bcrypt.hash(random, genSalt(8))
|
||||||
|
|
||||||
@ -51,6 +51,6 @@ proc makeSessionKey*(): string =
|
|||||||
# PROC : make password
|
# PROC : make password
|
||||||
## Creates an MD5 hash by combining password and salt
|
## Creates an MD5 hash by combining password and salt
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
proc makePassword*(password, salt: string, comparingTo = ""): string =
|
proc makePassword*(password, salt: string, comparingTo = ""): string =
|
||||||
let bcryptSalt = if comparingTo != "": comparingTo else: genSalt(8)
|
let bcryptSalt = if comparingTo != "": comparingTo else: genSalt(8)
|
||||||
result = hash(getMD5(salt & getMD5(password)), bcryptSalt)
|
result = hash(getMD5(salt & getMD5(password)), bcryptSalt)
|
||||||
|
22
code/utils.nim
Normal file
22
code/utils.nim
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
# --==--==--==--==--==--==--==--==--==--==--==--==--==--==--==-- #
|
||||||
|
# Bruno Charest
|
||||||
|
# 2022-10-25
|
||||||
|
#
|
||||||
|
# __ DESCRIPTIONS __
|
||||||
|
# utils : procedures generals utils
|
||||||
|
#
|
||||||
|
# --==--==--==--==--==--==--==--==--==--==--==--==--==--==--==-- #
|
||||||
|
|
||||||
|
import std/times
|
||||||
|
|
||||||
|
# --==--==--==--==--==--==--==--==--==--==--==--==--==--==-- #
|
||||||
|
# PROC : Convert epoch millisecond time to human readable time
|
||||||
|
# --==--==--==--==--==--==--==--==--==--==--==--==--==--==-- #
|
||||||
|
proc convertEpochToHumanTime*(epochTime: int): string =
|
||||||
|
var humanTime: Time
|
||||||
|
var dateTime: string
|
||||||
|
humanTime = fromUnix(convert(Milliseconds, Seconds, epochTime))
|
||||||
|
dateTime = format(humanTime, "yyyy-MM-dd' 'HH:mm")
|
||||||
|
# echo dateTime
|
||||||
|
return dateTime
|
||||||
|
|
@ -11,23 +11,38 @@ import joplin_utils
|
|||||||
|
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# TYPE : Selected Option for master web page
|
# TYPE : Selected Option for master web page
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
type
|
type
|
||||||
selectedOption* = enum
|
selectedOption* = enum
|
||||||
newNote="New Note", search="Search", shortcuts="Shortcuts", notes="Notes", notebooks="Notesbooks", tags="Tags"
|
newNote = "New Note", search = "Search", shortcuts = "Shortcuts",
|
||||||
|
notes = "Notes", notebooks = "Notesbooks", tags = "Tags",
|
||||||
|
notes_selectednote = "Notes-SelectedNote", dashbord = "Dashboard"
|
||||||
|
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# TYPE : Data Informations for Left Colomn
|
# TYPE : Data Informations for Left Colomn
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
type ColomnLeftData* = ref object of RootObj
|
type ColomnLeftData* = ref object of RootObj
|
||||||
j_status*: bool
|
j_status*: bool
|
||||||
option*: selectedOption
|
option*: selectedOption
|
||||||
j_notes*: joplin_notes
|
j_notes*: joplin_notes
|
||||||
j_notebooks*: joplin_notebooks
|
j_notes_nb*: int
|
||||||
j_tags*: joplin_tags
|
j_notebooks*: joplin_notebooks
|
||||||
req*: Request
|
j_notebooks_nb*: int
|
||||||
|
j_tags*: joplin_tags
|
||||||
|
j_tags_nb*: int
|
||||||
|
req*: Request
|
||||||
|
|
||||||
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
|
# TYPE : Data Informations for Right Colomn
|
||||||
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
|
type ColomnRightData* = ref object of RootObj
|
||||||
|
j_status*: bool
|
||||||
|
option*: selectedOption
|
||||||
|
j_SelectedNote*: joplin_note
|
||||||
|
j_notebooksNote*: joplin_notebooks
|
||||||
|
j_tagsNote*: joplin_tags
|
||||||
|
req*: Request
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -2,4 +2,5 @@ switch("d","release")
|
|||||||
# switch("d","ssl")
|
# switch("d","ssl")
|
||||||
# --threads:on
|
# --threads:on
|
||||||
--opt:size
|
--opt:size
|
||||||
switch("passL","-s")
|
switch("passL","-s")
|
||||||
|
hint("Name",false)
|
@ -26,7 +26,7 @@ port = "7000"
|
|||||||
#Joplin DESKTOP token
|
#Joplin DESKTOP token
|
||||||
#token = "e5f6644fbf6a97ddc55648dae72b11caecda6c6642d8ce0d3b20129b89b196385737eb908923542c3343649ebbf865b55bda031ab4c3a16edc7723ef2ad77d8f"
|
#token = "e5f6644fbf6a97ddc55648dae72b11caecda6c6642d8ce0d3b20129b89b196385737eb908923542c3343649ebbf865b55bda031ab4c3a16edc7723ef2ad77d8f"
|
||||||
# Joplin CLI token
|
# Joplin CLI token
|
||||||
token = "5b05f489016ce8a001ec83a7968419368eb9206340a18f73119c79e2154ab267ddec424658920bb6f88961c170a2680cd07fbd83f38e1c0c8df907a9aed22427"
|
token = "76d06ec328466c872f3a944f8237fd96f18d2b953ff013c8689304b7384f2f2232b3aedff079902217760e8fa180d1b89d2650ee1819dd628678fccdc0a140a6"
|
||||||
joplin_server = "https://joplinlab.bcmaison.cf"
|
joplin_server = "https://joplinlab.bcmaison.cf"
|
||||||
joplin_server_user = "joplinlab@zohomail.com"
|
joplin_server_user = "joplinlab@zohomail.com"
|
||||||
joplin_server_pwd = "Chab30017405"
|
joplin_server_pwd = "Chab30017405"
|
BIN
data/dashbord.xlsx
Normal file
BIN
data/dashbord.xlsx
Normal file
Binary file not shown.
1
data/toto.txt
Normal file
1
data/toto.txt
Normal file
@ -0,0 +1 @@
|
|||||||
|
SQLiteDatabase
|
@ -19,3 +19,15 @@ joplin config locale en_US
|
|||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
joplin config api.token
|
joplin config api.token
|
||||||
cat .config/joplin/settings.json
|
cat .config/joplin/settings.json
|
||||||
|
|
||||||
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
|
# install nim module requirment
|
||||||
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
|
nimble install jester
|
||||||
|
nimble install bcrypt
|
||||||
|
nimble install templates
|
||||||
|
|
||||||
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
|
# install packages requirment
|
||||||
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
|
sudo apt install libzip4
|
175
main.nim
175
main.nim
@ -16,20 +16,21 @@
|
|||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# Import section
|
# Import section
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
import os # Used to get arguments
|
import os # Used to get arguments
|
||||||
import uri # We need to encode urls: encodeUrl()
|
import uri # We need to encode urls: encodeUrl()
|
||||||
import times # Time and date
|
import xlsx # read dashbord.xlsx
|
||||||
import jester # Our webserver
|
import times # Time and date
|
||||||
import logging # Logging utils
|
import jester # Our webserver
|
||||||
import strutils # Basic functions
|
import logging # Logging utils
|
||||||
import parsecfg # Parse CFG (config) files
|
import strutils # Basic functions
|
||||||
import std/json # json manipulation
|
import parsecfg # Parse CFG (config) files
|
||||||
|
import std/json # json manipulation
|
||||||
import db_sqlite # SQLite
|
import db_sqlite # SQLite
|
||||||
|
|
||||||
import code/database_utils # Utils used in the database
|
import code/database_utils # Utils used in the database
|
||||||
import code/password_utils # Our file with password utils
|
import code/password_utils # Our file with password utils
|
||||||
import code/joplin_utils # Joplin utils procedures and types
|
import code/joplin_utils # Joplin utils procedures and types
|
||||||
import code/web_utils # Web utils procedures and types
|
import code/web_utils # Web utils procedures and types
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# First we'll load config files
|
# First we'll load config files
|
||||||
@ -41,14 +42,14 @@ let dict = loadConfig("config/config.cfg")
|
|||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
|
|
||||||
# Database
|
# Database
|
||||||
let db_user = dict.getSectionValue("Database", "user")
|
let db_user = dict.getSectionValue("Database", "user")
|
||||||
let db_pass = dict.getSectionValue("Database", "pass")
|
let db_pass = dict.getSectionValue("Database", "pass")
|
||||||
let db_name = dict.getSectionValue("Database", "name")
|
let db_name = dict.getSectionValue("Database", "name")
|
||||||
let db_host = dict.getSectionValue("Database", "host")
|
let db_host = dict.getSectionValue("Database", "host")
|
||||||
|
|
||||||
# Website
|
# Website
|
||||||
let mainURL = dict.getSectionValue("Server", "url")
|
let mainURL = dict.getSectionValue("Server", "url")
|
||||||
let mainPort = parseInt dict.getSectionValue("Server", "port")
|
let mainPort = parseInt dict.getSectionValue("Server", "port")
|
||||||
let mainWebsite = dict.getSectionValue("Server", "website")
|
let mainWebsite = dict.getSectionValue("Server", "website")
|
||||||
|
|
||||||
# Joplin
|
# Joplin
|
||||||
@ -79,27 +80,23 @@ type
|
|||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# proc init : initialisation variables
|
# proc init : initialisation variables
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
proc init(c: var TData, cld: var ColomnLeftData) =
|
proc init(c: var TData, cld: var ColomnLeftData, crd: var ColomnRightData) =
|
||||||
## Empty out user session data
|
## Empty out user session data
|
||||||
c.userpass = ""
|
c.userpass = ""
|
||||||
c.username = ""
|
c.username = ""
|
||||||
c.userid = ""
|
c.userid = ""
|
||||||
c.loggedIn = false
|
c.loggedIn = false
|
||||||
c.notification = 0
|
c.notification = 0
|
||||||
|
|
||||||
|
|
||||||
## default option
|
|
||||||
# cld.option = notes
|
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# function : loggedIn
|
# function : loggedIn
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
func loggedIn(c: TData): bool =
|
func loggedIn(c: TData): bool =
|
||||||
## Check if user is logged in by verifying that c.username exists
|
## Check if user is logged in by verifying that c.username exists
|
||||||
c.username.len > 0
|
c.username.len > 0
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# proc checkLoggedIn
|
# proc checkLoggedIn
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
proc checkLoggedIn(c: var TData) =
|
proc checkLoggedIn(c: var TData) =
|
||||||
## Check if user is logged in
|
## Check if user is logged in
|
||||||
@ -113,21 +110,24 @@ proc checkLoggedIn(c: var TData) =
|
|||||||
# Update the value lastModified for the user in the
|
# Update the value lastModified for the user in the
|
||||||
# table session where the sid and IP match. If there's
|
# table session where the sid and IP match. If there's
|
||||||
# any results (above 0) assign values
|
# any results (above 0) assign values
|
||||||
if execAffectedRows(db, sql("UPDATE session SET lastModified = " & $toInt(epochTime()) & " " & "WHERE ip = ? AND key = ?"), c.req.ip, sid) > 0:
|
if execAffectedRows(db, sql("UPDATE session SET lastModified = " & $toInt(
|
||||||
|
epochTime()) & " " & "WHERE ip = ? AND key = ?"), c.req.ip, sid) > 0:
|
||||||
|
|
||||||
# Get user data based on userID from session table
|
# Get user data based on userID from session table
|
||||||
# Assign values to user details - `c`
|
# Assign values to user details - `c`
|
||||||
c.userid = getValue(db, sql"SELECT userid FROM session WHERE ip = ? AND key = ?", c.req.ip, sid)
|
c.userid = getValue(db, sql"SELECT userid FROM session WHERE ip = ? AND key = ?",
|
||||||
|
c.req.ip, sid)
|
||||||
|
|
||||||
# Get user data based on userID from person table
|
# Get user data based on userID from person table
|
||||||
let row = getRow(db, sql"SELECT name, email, status FROM person WHERE id = ?", c.userid)
|
let row = getRow(db, sql"SELECT name, email, status FROM person WHERE id = ?", c.userid)
|
||||||
|
|
||||||
# Assign user data
|
# Assign user data
|
||||||
c.username = row[0]
|
c.username = row[0]
|
||||||
c.email = toLowerAscii(row[1])
|
c.email = toLowerAscii(row[1])
|
||||||
|
|
||||||
# Update our session table with info about activity
|
# Update our session table with info about activity
|
||||||
discard tryExec(db, sql"UPDATE person SET lastOnline = ? WHERE id = ?", toInt(epochTime()), c.userid)
|
discard tryExec(db, sql"UPDATE person SET lastOnline = ? WHERE id = ?",
|
||||||
|
toInt(epochTime()), c.userid)
|
||||||
|
|
||||||
else:
|
else:
|
||||||
# If the user is not found in the session table
|
# If the user is not found in the session table
|
||||||
@ -158,14 +158,15 @@ proc login(c: var TData, email, pass: string): tuple[b: bool, s: string] =
|
|||||||
# and salt.
|
# and salt.
|
||||||
if row[2] == makePassword(pass, row[4], row[2]):
|
if row[2] == makePassword(pass, row[4], row[2]):
|
||||||
# Assign the values
|
# Assign the values
|
||||||
c.userid = row[0]
|
c.userid = row[0]
|
||||||
c.username = row[1]
|
c.username = row[1]
|
||||||
c.userpass = row[2]
|
c.userpass = row[2]
|
||||||
c.email = toLowerAscii(row[3])
|
c.email = toLowerAscii(row[3])
|
||||||
|
|
||||||
# Generate session key and save it
|
# Generate session key and save it
|
||||||
let key = makeSessionKey()
|
let key = makeSessionKey()
|
||||||
exec(db, sql"INSERT INTO session (ip, key, userid) VALUES (?, ?, ?)", c.req.ip, key, row[0])
|
exec(db, sql"INSERT INTO session (ip, key, userid) VALUES (?, ?, ?)",
|
||||||
|
c.req.ip, key, row[0])
|
||||||
|
|
||||||
info("Login successful")
|
info("Login successful")
|
||||||
return (true, key)
|
return (true, key)
|
||||||
@ -196,16 +197,19 @@ template createTFD() =
|
|||||||
|
|
||||||
# Assign the c to TDATA
|
# Assign the c to TDATA
|
||||||
var cld {.inject.}: ColomnLeftData
|
var cld {.inject.}: ColomnLeftData
|
||||||
|
var crd {.inject.}: ColomnRightData
|
||||||
|
|
||||||
# New instance of c
|
# New instance of c
|
||||||
new(c)
|
new(c)
|
||||||
new(cld)
|
new(cld)
|
||||||
|
new(crd)
|
||||||
|
|
||||||
# Set standard values
|
# Set standard values
|
||||||
init(c,cld)
|
init(c, cld, crd)
|
||||||
# Get users request
|
# Get users request
|
||||||
c.req = request
|
c.req = request
|
||||||
cld.req = request
|
cld.req = request
|
||||||
|
crd.req = request
|
||||||
|
|
||||||
# Check for cookies (we need the cookie named sid)
|
# Check for cookies (we need the cookie named sid)
|
||||||
if cookies(request).len > 0:
|
if cookies(request).len > 0:
|
||||||
@ -229,7 +233,8 @@ when isMainModule:
|
|||||||
# Connect to DB
|
# Connect to DB
|
||||||
try:
|
try:
|
||||||
# We are using the values which we assigned earlier
|
# We are using the values which we assigned earlier
|
||||||
db = open(connection=db_host, user=db_user, password=db_pass, database=db_name)
|
db = open(connection = db_host, user = db_user, password = db_pass,
|
||||||
|
database = db_name)
|
||||||
info("Connection to DB is established.")
|
info("Connection to DB is established.")
|
||||||
except:
|
except:
|
||||||
fatal("Connection to DB could not be established.")
|
fatal("Connection to DB could not be established.")
|
||||||
@ -262,7 +267,7 @@ include "tmpl/tests/test_viewtree.tmpl"
|
|||||||
# Setup routes (URL's)
|
# Setup routes (URL's)
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
routes:
|
routes:
|
||||||
|
|
||||||
# default route
|
# default route
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
get "/":
|
get "/":
|
||||||
@ -274,20 +279,30 @@ routes:
|
|||||||
get "/secret":
|
get "/secret":
|
||||||
createTFD()
|
createTFD()
|
||||||
echo c.loggedIn
|
echo c.loggedIn
|
||||||
echo @"msg"
|
var isStart: int = 0
|
||||||
if c.loggedIn:
|
|
||||||
|
|
||||||
# Start joplin terminal cli if stropped
|
|
||||||
if @"msg" == "startStopJoplin":
|
# if joplin_cli_status() == false:
|
||||||
if joplin_cli_status() == false:
|
# isStart = joplin_cli_start()
|
||||||
var isStart = joplin_cli_start()
|
|
||||||
echo "Joplin client Terminal started: ",isStart
|
|
||||||
# echo joplin_cli_status()
|
echo "MESSAGE :", @"msg"
|
||||||
if joplin_cli_status() == true:
|
echo "NOTE_ID :", @"noteid"
|
||||||
var isStart = joplin_cli_stop()
|
let selectedNoteId = @"noteid"
|
||||||
echo "Joplin client Terminal stopped: ",isStart
|
var url_note = "/secret?msg=notes"
|
||||||
# echo joplin_cli_status()
|
|
||||||
redirect("/secret")
|
|
||||||
|
# URL msg and note id if exist noteid
|
||||||
|
if selectedNoteId != "":
|
||||||
|
url_note = "/secret?msg=notes¬eid=" & selectedNoteId
|
||||||
|
|
||||||
|
if c.loggedIn:
|
||||||
|
# Start joplin terminal cli if stropped
|
||||||
|
if @"msg" == "startStopJoplin":
|
||||||
|
var isStart = joplin_cli_start_stop()
|
||||||
|
|
||||||
|
|
||||||
|
redirect(url_note)
|
||||||
|
|
||||||
# if Joplin application work
|
# if Joplin application work
|
||||||
var checkJoplin = waitFor ping_joplin(joplin_token)
|
var checkJoplin = waitFor ping_joplin(joplin_token)
|
||||||
@ -299,51 +314,81 @@ routes:
|
|||||||
# determine the section to uptade
|
# determine the section to uptade
|
||||||
if @"msg" == "newNote":
|
if @"msg" == "newNote":
|
||||||
cld.option = newNote
|
cld.option = newNote
|
||||||
echo "Todo"
|
echo "=> Section newNote"
|
||||||
|
|
||||||
elif @"msg" == "search":
|
elif @"msg" == "search":
|
||||||
cld.option = search
|
cld.option = search
|
||||||
echo "Todo"
|
echo "=> Section search"
|
||||||
|
|
||||||
elif @"msg" == "shortcuts":
|
elif @"msg" == "shortcuts":
|
||||||
cld.option = shortcuts
|
cld.option = shortcuts
|
||||||
echo "Todo"
|
echo "=> Section shortcuts"
|
||||||
|
|
||||||
elif @"msg" == "notebooks":
|
elif @"msg" == "notebooks":
|
||||||
|
echo "=> Section notebooks"
|
||||||
cld.option = notebooks
|
cld.option = notebooks
|
||||||
cld.j_notebooks = waitFor get_joplin_notebooks(joplin_token)
|
cld.j_notebooks = waitFor get_joplin_notebooks(joplin_token)
|
||||||
|
|
||||||
elif @"msg" == "notes":
|
elif @"msg" == "notes":
|
||||||
|
echo "=> Section notes"
|
||||||
cld.option = notes
|
cld.option = notes
|
||||||
cld.j_notes = waitFor get_joplin_notes(joplin_token)
|
cld.j_notes = waitFor get_joplin_notes(joplin_token)
|
||||||
|
cld.j_notes_nb = cld.j_notes.id.len()
|
||||||
|
|
||||||
|
# for i in 0 .. (cld.j_notes.id.len() - 1):
|
||||||
|
# echo cld.j_notes.id[i]
|
||||||
|
# echo cld.j_notes.created_time[i]
|
||||||
|
|
||||||
|
if selectedNoteId != "":
|
||||||
|
crd.j_SelectedNote = waitFor get_joplin_note(joplin_token, selectedNoteId)
|
||||||
|
|
||||||
elif @"msg" == "tags":
|
elif @"msg" == "tags":
|
||||||
|
echo "=> Section tags"
|
||||||
cld.option = tags
|
cld.option = tags
|
||||||
cld.j_tags = waitFor get_joplin_tags(joplin_token)
|
cld.j_tags = waitFor get_joplin_tags(joplin_token)
|
||||||
|
|
||||||
|
elif @"msg" == "dashbord":
|
||||||
|
echo "=> Section dashbord"
|
||||||
|
#cld.option = dashbord
|
||||||
|
|
||||||
|
let
|
||||||
|
data = parseExcel("data/dashbord.xlsx", header = true)
|
||||||
|
sheetName = "dashbord"
|
||||||
|
echo data[sheetName]
|
||||||
|
|
||||||
|
let rows = data[sheetName].toSeq(true)
|
||||||
|
for row in rows:
|
||||||
|
echo "position: " & row[0]
|
||||||
|
echo "URL externe: " & row[1]
|
||||||
|
|
||||||
|
|
||||||
elif @"msg" == "sendFeedBack":
|
elif @"msg" == "sendFeedBack":
|
||||||
echo "Todo"
|
echo "Todo"
|
||||||
|
|
||||||
resp Http200, {"Access-Control-Allow-Origin": "http://127.0.0.1:7000"}, genSecret(c,cld)
|
resp Http200, {"Access-Control-Allow-Origin": "http://127.0.0.1:7000"},
|
||||||
|
genSecret(c, cld, crd)
|
||||||
|
|
||||||
|
|
||||||
# Login route
|
# Login route
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
get "/login":
|
get "/login":
|
||||||
createTFD()
|
createTFD()
|
||||||
resp Http200, {"Access-Control-Allow-Origin": "http://127.0.0.1:7000"}, genLogin(c, @"msg")
|
resp Http200, {"Access-Control-Allow-Origin": "http://127.0.0.1:7000"},
|
||||||
|
genLogin(c, @"msg")
|
||||||
|
|
||||||
# action route during login
|
# action route during login
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
post "/dologin":
|
post "/dologin":
|
||||||
createTFD()
|
createTFD()
|
||||||
|
|
||||||
let (loginB, loginS) = login(c, replace(toLowerAscii(@"email"), " ", ""), replace(@"password", " ", ""))
|
let (loginB, loginS) = login(c, replace(toLowerAscii(@"email"), " ", ""),
|
||||||
|
replace(@"password", " ", ""))
|
||||||
if loginB:
|
if loginB:
|
||||||
when defined(dev):
|
when defined(dev):
|
||||||
jester.setCookie("sid", loginS, daysForward(7))
|
jester.setCookie("sid", loginS, daysForward(7))
|
||||||
else:
|
else:
|
||||||
jester.setCookie("sid", loginS, daysForward(7), samesite = Lax, secure = true, httpOnly = true)
|
jester.setCookie("sid", loginS, daysForward(7), samesite = Lax,
|
||||||
|
secure = true, httpOnly = true)
|
||||||
#jester.setCookie("sid", loginS, daysForward(7), samesite = Lax, secure = true, httpOnly = true)
|
#jester.setCookie("sid", loginS, daysForward(7), samesite = Lax, secure = true, httpOnly = true)
|
||||||
|
|
||||||
redirect("/secret")
|
redirect("/secret")
|
||||||
@ -363,7 +408,7 @@ routes:
|
|||||||
# if joplin_cli_status() == false:
|
# if joplin_cli_status() == false:
|
||||||
# joplin_cli_start()
|
# joplin_cli_start()
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
@ -380,22 +425,22 @@ routes:
|
|||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
get "/test_pingjoplin":
|
get "/test_pingjoplin":
|
||||||
createTFD()
|
createTFD()
|
||||||
|
|
||||||
var pingCheck: joplin_ping
|
var pingCheck: joplin_ping
|
||||||
pingCheck = waitFor ping_joplin(joplin_token)
|
pingCheck = waitFor ping_joplin(joplin_token)
|
||||||
echo pingCheck.ping_status
|
echo pingCheck.ping_status
|
||||||
|
|
||||||
resp test_ping(c, pingCheck)
|
resp test_ping(c, pingCheck)
|
||||||
|
|
||||||
# Test geting list of all notebooks
|
# Test geting list of all notebooks
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
get "/test_notebooks":
|
get "/test_notebooks":
|
||||||
createTFD()
|
createTFD()
|
||||||
cld.j_notebooks = waitFor get_joplin_notebooks(joplin_token)
|
cld.j_notebooks = waitFor get_joplin_notebooks(joplin_token)
|
||||||
resp test_notebooks(c, cld)
|
resp test_notebooks(c, cld)
|
||||||
|
|
||||||
# Test geting list of all notes
|
# Test geting list of all notes
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
get "/test_notes":
|
get "/test_notes":
|
||||||
createTFD()
|
createTFD()
|
||||||
cld.j_notes = waitFor get_joplin_notes(joplin_token)
|
cld.j_notes = waitFor get_joplin_notes(joplin_token)
|
||||||
@ -408,13 +453,13 @@ routes:
|
|||||||
cld.j_tags = waitFor get_joplin_tags(joplin_token)
|
cld.j_tags = waitFor get_joplin_tags(joplin_token)
|
||||||
resp test_tags(c, cld)
|
resp test_tags(c, cld)
|
||||||
|
|
||||||
# Test a viewtree
|
# Test a viewtree
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
get "/test_viewtree":
|
get "/test_viewtree":
|
||||||
createTFD()
|
createTFD()
|
||||||
resp test_viewtree(c)
|
resp test_viewtree(c)
|
||||||
|
|
||||||
# Test geting all tags as JSON output
|
# Test geting all tags as JSON output
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
get "/test_tags_json":
|
get "/test_tags_json":
|
||||||
createTFD()
|
createTFD()
|
||||||
@ -426,4 +471,4 @@ routes:
|
|||||||
|
|
||||||
# # ##
|
# # ##
|
||||||
# # END TESTS SECTION ##
|
# # END TESTS SECTION ##
|
||||||
# # ##
|
# # ##
|
||||||
|
@ -2,449 +2,493 @@
|
|||||||
[ FONT ]*/
|
[ FONT ]*/
|
||||||
|
|
||||||
@font-face {
|
@font-face {
|
||||||
font-family: Poppins-Regular;
|
font-family: Poppins-Regular;
|
||||||
src: url('../fonts/poppins/Poppins-Regular.ttf');
|
src: url('../fonts/poppins/Poppins-Regular.ttf');
|
||||||
}
|
}
|
||||||
|
|
||||||
@font-face {
|
@font-face {
|
||||||
font-family: Poppins-Medium;
|
font-family: Poppins-Medium;
|
||||||
src: url('../fonts/poppins/Poppins-Medium.ttf');
|
src: url('../fonts/poppins/Poppins-Medium.ttf');
|
||||||
}
|
}
|
||||||
|
|
||||||
@font-face {
|
@font-face {
|
||||||
font-family: Poppins-Bold;
|
font-family: Poppins-Bold;
|
||||||
src: url('../fonts/poppins/Poppins-Bold.ttf');
|
src: url('../fonts/poppins/Poppins-Bold.ttf');
|
||||||
}
|
}
|
||||||
|
|
||||||
@font-face {
|
@font-face {
|
||||||
font-family: Poppins-SemiBold;
|
font-family: Poppins-SemiBold;
|
||||||
src: url('../fonts/poppins/Poppins-SemiBold.ttf');
|
src: url('../fonts/poppins/Poppins-SemiBold.ttf');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
html,
|
||||||
html, body {
|
body {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
font-family: Poppins-Regular, sans-serif;
|
font-family: Poppins-Regular, sans-serif;
|
||||||
height:100%;
|
height: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
*,
|
*,
|
||||||
*::before,
|
*::before,
|
||||||
*::after {
|
*::after {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
margin: 0;
|
margin: 0;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
a {
|
a {
|
||||||
font-family: Poppins-Regular;
|
font-family: Poppins-Regular;
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
line-height: 1.7;
|
line-height: 1.7;
|
||||||
color: #666666;
|
color: #666666;
|
||||||
margin: 0px;
|
margin: 0px;
|
||||||
transition: all 0.4s;
|
transition: all 0.4s;
|
||||||
-webkit-transition: all 0.4s;
|
-webkit-transition: all 0.4s;
|
||||||
-o-transition: all 0.4s;
|
-o-transition: all 0.4s;
|
||||||
-moz-transition: all 0.4s;
|
-moz-transition: all 0.4s;
|
||||||
}
|
}
|
||||||
|
|
||||||
:root {
|
:root {
|
||||||
--accent-color: #0053b8;
|
--accent-color: #0053b8;
|
||||||
--lightest-gray: rgb(244, 244, 244);
|
--lightest-gray: rgb(244, 244, 244);
|
||||||
--light-gray: rgb(144, 144, 144);
|
--light-gray: rgb(144, 144, 144);
|
||||||
--medium-gray: rgb(96, 96, 96);
|
--medium-gray: rgb(96, 96, 96);
|
||||||
--dark-gray: rgb(13, 13, 13);
|
--dark-gray: rgb(13, 13, 13);
|
||||||
--animation-duration: 200ms;
|
--animation-duration: 200ms;
|
||||||
--animation-timing-curve: ease-in-out;
|
--animation-timing-curve: ease-in-out;
|
||||||
--blue-joplin-color: #0053b8;
|
--blue-joplin-color: #0053b8;
|
||||||
--light-blue-joplin-color: rgb(237, 241, 243);
|
/* --blue-joplin-color: #b83100; */
|
||||||
--header-height: 55px;
|
--light-blue-joplin-color: rgb(237, 241, 243);
|
||||||
|
--header-height: 55px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.header {
|
.header {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
position: sticky;
|
position: sticky;
|
||||||
top: 0;
|
top: 0;
|
||||||
background-color: white;
|
background-color: white;
|
||||||
box-shadow: 0 1px 10px 0 rgba(0, 0, 0, 0.4);
|
box-shadow: 0 1px 10px 0 rgba(0, 0, 0, 0.4);
|
||||||
padding: 0 0.5rem;
|
padding: 0 0.5rem;
|
||||||
height: var(--header-height);
|
height: var(--header-height);
|
||||||
}
|
}
|
||||||
|
|
||||||
.login {
|
.login {
|
||||||
font-size: large;
|
font-size: large;
|
||||||
fill: var(--blue-joplin-color);
|
fill: var(--blue-joplin-color);
|
||||||
/* max-width: fit-content; */
|
/* max-width: fit-content; */
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: 30%;
|
top: 30%;
|
||||||
left: 50%;
|
left: 50%;
|
||||||
transform: translate(-52%, -50%);
|
transform: translate(-52%, -50%);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* Create two equal columns that floats next to each other */
|
/* Create two equal columns that floats next to each other */
|
||||||
|
|
||||||
.column {
|
.column {
|
||||||
float: left;
|
float: left;
|
||||||
padding: 10px;
|
padding: 10px;
|
||||||
height: calc(100vh - var(--header-height));
|
height: calc(100vh - var(--header-height));
|
||||||
/* height: auto; */
|
/* height: auto; */
|
||||||
/* flex: 50%; */
|
/* flex: 50%; */
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.column .scrollbar {
|
.column .scrollbar {
|
||||||
/* margin-left: 30px; */
|
/* margin-left: 30px; */
|
||||||
/* float: left; */
|
/* float: left; */
|
||||||
/* min-height: 100px; */
|
/* min-height: 100px; */
|
||||||
height: 800px;
|
height: 800px;
|
||||||
background: #fff;
|
background: #fff;
|
||||||
overflow-y: scroll;
|
overflow-y: scroll;
|
||||||
/* margin-bottom: 25px; */
|
/* margin-bottom: 25px; */
|
||||||
}
|
}
|
||||||
|
|
||||||
.left {
|
.left {
|
||||||
width: 25%;
|
width: 25%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.right {
|
.right {
|
||||||
width: 75%;
|
width: 75%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.row {
|
.row {
|
||||||
display: flex;
|
display: flex;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* Clear floats after the columns */
|
/* Clear floats after the columns */
|
||||||
|
|
||||||
.row:after {
|
.row:after {
|
||||||
content: "";
|
content: "";
|
||||||
display: table;
|
display: table;
|
||||||
clear: both;
|
clear: both;
|
||||||
}
|
}
|
||||||
|
|
||||||
.menu-icon-btn {
|
.menu-icon-btn {
|
||||||
background: none;
|
background: none;
|
||||||
border: none;
|
border: none;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.notifBell-btn {
|
.notifBell-btn {
|
||||||
background: none;
|
background: none;
|
||||||
border: none;
|
border: none;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.menu-icon {
|
.menu-icon {
|
||||||
width: 25px;
|
width: 25px;
|
||||||
height: 25px;
|
height: 25px;
|
||||||
fill: var(--medium-gray);
|
fill: var(--medium-gray);
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
.menu-icon:hover {
|
.menu-icon:hover {
|
||||||
fill: var(--dark-gray);
|
fill: var(--dark-gray);
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar {
|
.sidebar {
|
||||||
flex-shrink: 0;
|
flex-shrink: 0;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
width: 75px;
|
width: 75px;
|
||||||
border-right: 1px solid var(--light-gray);
|
border-right: 1px solid var(--light-gray);
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
height: calc(100vh - var(--header-height));
|
height: calc(100vh - var(--header-height));
|
||||||
padding-top: 1rem;
|
padding-top: 1rem;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: stretch;
|
justify-content: stretch;
|
||||||
transition: width var(--animation-duration) var(--animation-timing-curve);
|
transition: width var(--animation-duration) var(--animation-timing-curve);
|
||||||
position: sticky;
|
position: sticky;
|
||||||
left: 0;
|
left: 0;
|
||||||
top: var(--header-height);
|
top: var(--header-height);
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar .hidden-sidebar {
|
.sidebar .hidden-sidebar {
|
||||||
opacity: 0;
|
opacity: 0;
|
||||||
width: 0;
|
width: 0;
|
||||||
transition: opacity var(--animation-duration) var(--animation-timing-curve);
|
transition: opacity var(--animation-duration) var(--animation-timing-curve);
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar.open .hidden-sidebar {
|
.sidebar.open .hidden-sidebar {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: auto;
|
height: auto;
|
||||||
opacity: 1;
|
opacity: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar .top-sidebar {
|
.sidebar .top-sidebar {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar .channel-logo {
|
.sidebar .channel-logo {
|
||||||
display: block;
|
display: block;
|
||||||
width: 30px;
|
width: 30px;
|
||||||
height: 30px;
|
height: 30px;
|
||||||
transition: var(--animation-duration) var(--animation-timing-curve);
|
transition: var(--animation-duration) var(--animation-timing-curve);
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar.open .channel-logo {
|
.sidebar.open .channel-logo {
|
||||||
width: 90px;
|
width: 90px;
|
||||||
height: 90px;
|
height: 90px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar .channel-logo>img {
|
.sidebar .channel-logo>img {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.middle-sidebar {
|
.middle-sidebar {
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
overflow-x: hidden;
|
overflow-x: hidden;
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
margin: 1rem 0;
|
margin: 1rem 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.middle-sidebar,
|
.middle-sidebar,
|
||||||
.bottom-sidebar {
|
.bottom-sidebar {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.container {
|
.container {
|
||||||
display: flex;
|
display: flex;
|
||||||
height: 94%;
|
height: 94%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.vertical-center {
|
.vertical-center {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: 50%;
|
top: 50%;
|
||||||
-ms-transform: translateY(-50%);
|
-ms-transform: translateY(-50%);
|
||||||
transform: translateY(-50%);
|
transform: translateY(-50%);
|
||||||
/* border: 5px solid #FFFF00; */
|
/* border: 5px solid #FFFF00; */
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
.content {
|
.content {
|
||||||
/* margin: 1rem; */
|
/* margin: 1rem; */
|
||||||
width: 1200px;
|
width: 1200px;
|
||||||
min-height: 100px;
|
min-height: 100px;
|
||||||
height: auto;
|
height: auto;
|
||||||
margin: 80px auto 0px auto;
|
margin: 80px auto 0px auto;
|
||||||
/* background-color: black; */
|
/* background-color: black; */
|
||||||
padding: 10px 10px;
|
padding: 10px 10px;
|
||||||
display:table;
|
display: table;
|
||||||
overflow:hidden;
|
overflow: hidden;
|
||||||
height:100%;
|
height: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar-list {
|
.sidebar-list {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
list-style: none;
|
list-style: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar.open .sidebar-link {
|
.sidebar.open .sidebar-link {
|
||||||
justify-content: flex-start;
|
justify-content: flex-start;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.svg-icon path,
|
||||||
|
.svg-icon polygon,
|
||||||
|
.svg-icon rect,
|
||||||
.sidebar-icon {
|
.sidebar-icon {
|
||||||
width: 25px;
|
width: 25px;
|
||||||
height: 25px;
|
height: 25px;
|
||||||
flex-shrink: 0;
|
flex-shrink: 0;
|
||||||
fill: var(--blue-joplin-color);
|
fill: var(--blue-joplin-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
.svg-icon {
|
||||||
|
width: 25px;
|
||||||
|
height: 25px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.svg-icon circle {
|
||||||
|
stroke: #4691f6;
|
||||||
|
stroke-width: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar-list .hidden-sidebar {
|
.sidebar-list .hidden-sidebar {
|
||||||
margin-left: 1.5rem;
|
margin-left: 1.5rem;
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar-link {
|
.sidebar-link {
|
||||||
display: flex;
|
display: flex;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
padding: 0.5rem 0;
|
padding: 0.5rem 0;
|
||||||
color: var(--light-gray);
|
color: var(--light-gray);
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
padding-left: 25px;
|
padding-left: 25px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sidebar-note-link {
|
||||||
|
display: flex;
|
||||||
|
width: 100%;
|
||||||
|
/* padding: 10px; */
|
||||||
|
color: var(--blue-joplin-color);
|
||||||
|
text-decoration: none;
|
||||||
|
font-size: 16px;
|
||||||
|
align-items: left;
|
||||||
|
padding-left: 15px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sidebar-note-date {
|
||||||
|
display: flex;
|
||||||
|
width: 100%;
|
||||||
|
/* padding: 0.1rem 0; */
|
||||||
|
color: var(--light-gray);
|
||||||
|
text-decoration: none;
|
||||||
|
font-size: 12px;
|
||||||
|
align-items: left;
|
||||||
|
padding-left: 15px;
|
||||||
|
padding-bottom: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar-list-item {
|
.sidebar-list-item {
|
||||||
position: relative;
|
position: relative;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
fill: var(--light-gray);
|
fill: var(--light-gray);
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar-list-item.active {
|
.sidebar-list-item.active {
|
||||||
fill: var(--accent-color);
|
fill: var(--accent-color);
|
||||||
background-color: var(--light-blue-joplin-color);
|
background-color: var(--light-blue-joplin-color);
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar-list-item:hover {
|
.sidebar-list-item:hover {
|
||||||
background-color: var(--light-blue-joplin-color);
|
background-color: var(--light-blue-joplin-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
.columnLeftJoplinNotes:hover {
|
||||||
|
background-color: var(--light-blue-joplin-color);
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar-list-item.active::before {
|
.sidebar-list-item.active::before {
|
||||||
content: "";
|
content: "";
|
||||||
background-color: var(--accent-color);
|
background-color: var(--accent-color);
|
||||||
height: 100%;
|
height: 100%;
|
||||||
left: 0;
|
left: 0;
|
||||||
width: 3px;
|
width: 3px;
|
||||||
position: absolute;
|
position: absolute;
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar.open {
|
.sidebar.open {
|
||||||
width: 200px;
|
width: 200px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.your-channel {
|
.your-channel {
|
||||||
color: var(--dark-gray);
|
color: var(--dark-gray);
|
||||||
font-size: 1.5rem;
|
font-size: 1.5rem;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
margin-bottom: 0.15rem;
|
margin-bottom: 0.15rem;
|
||||||
margin-top: 0.5rem;
|
margin-top: 0.5rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.channel-name {
|
.channel-name {
|
||||||
color: var(--medium-gray);
|
color: var(--medium-gray);
|
||||||
font-size: 1.0rem;
|
font-size: 1.0rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar .top-sidebar {
|
.sidebar .top-sidebar {
|
||||||
height: 30px;
|
height: 30px;
|
||||||
transition: height var(--animation-duration) var(--animation-timing-curve);
|
transition: height var(--animation-duration) var(--animation-timing-curve);
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar.open .top-sidebar {
|
.sidebar.open .top-sidebar {
|
||||||
height: 150px;
|
height: 150px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.sidebar .top-sidebar .hidden-sidebar {
|
.sidebar .top-sidebar .hidden-sidebar {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.force-overflow {
|
.force-overflow {
|
||||||
min-height: 450px;
|
min-height: 450px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scrollbar-primary::-webkit-scrollbar {
|
.scrollbar-primary::-webkit-scrollbar {
|
||||||
width: 6px;
|
width: 6px;
|
||||||
background-color: #F5F5F5;
|
background-color: #F5F5F5;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scrollbar-primary::-webkit-scrollbar-thumb {
|
.scrollbar-primary::-webkit-scrollbar-thumb {
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
||||||
background-color: #4285F4;
|
background-color: #4285F4;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scrollbar-danger::-webkit-scrollbar-track {
|
.scrollbar-danger::-webkit-scrollbar-track {
|
||||||
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
||||||
background-color: #F5F5F5;
|
background-color: #F5F5F5;
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scrollbar-danger::-webkit-scrollbar {
|
.scrollbar-danger::-webkit-scrollbar {
|
||||||
width: 12px;
|
width: 12px;
|
||||||
background-color: #F5F5F5;
|
background-color: #F5F5F5;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scrollbar-danger::-webkit-scrollbar-thumb {
|
.scrollbar-danger::-webkit-scrollbar-thumb {
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
||||||
background-color: #ff3547;
|
background-color: #ff3547;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scrollbar-warning::-webkit-scrollbar-track {
|
.scrollbar-warning::-webkit-scrollbar-track {
|
||||||
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
||||||
background-color: #F5F5F5;
|
background-color: #F5F5F5;
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scrollbar-warning::-webkit-scrollbar {
|
.scrollbar-warning::-webkit-scrollbar {
|
||||||
width: 12px;
|
width: 12px;
|
||||||
background-color: #F5F5F5;
|
background-color: #F5F5F5;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scrollbar-warning::-webkit-scrollbar-thumb {
|
.scrollbar-warning::-webkit-scrollbar-thumb {
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
||||||
background-color: #FF8800;
|
background-color: #FF8800;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scrollbar-success::-webkit-scrollbar-track {
|
.scrollbar-success::-webkit-scrollbar-track {
|
||||||
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
||||||
background-color: #F5F5F5;
|
background-color: #F5F5F5;
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scrollbar-success::-webkit-scrollbar {
|
.scrollbar-success::-webkit-scrollbar {
|
||||||
width: 12px;
|
width: 12px;
|
||||||
background-color: #F5F5F5;
|
background-color: #F5F5F5;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scrollbar-success::-webkit-scrollbar-thumb {
|
.scrollbar-success::-webkit-scrollbar-thumb {
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
||||||
background-color: #00C851;
|
background-color: #00C851;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scrollbar-info::-webkit-scrollbar-track {
|
.scrollbar-info::-webkit-scrollbar-track {
|
||||||
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
||||||
background-color: #F5F5F5;
|
background-color: #F5F5F5;
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scrollbar-info::-webkit-scrollbar {
|
.scrollbar-info::-webkit-scrollbar {
|
||||||
width: 12px;
|
width: 12px;
|
||||||
background-color: #F5F5F5;
|
background-color: #F5F5F5;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scrollbar-info::-webkit-scrollbar-thumb {
|
.scrollbar-info::-webkit-scrollbar-thumb {
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
||||||
background-color: #33b5e5;
|
background-color: #33b5e5;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scrollbar-default::-webkit-scrollbar-track {
|
.scrollbar-default::-webkit-scrollbar-track {
|
||||||
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
||||||
background-color: #F5F5F5;
|
background-color: #F5F5F5;
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scrollbar-default::-webkit-scrollbar {
|
.scrollbar-default::-webkit-scrollbar {
|
||||||
width: 12px;
|
width: 12px;
|
||||||
background-color: #F5F5F5;
|
background-color: #F5F5F5;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scrollbar-default::-webkit-scrollbar-thumb {
|
.scrollbar-default::-webkit-scrollbar-thumb {
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
||||||
background-color: #2BBBAD;
|
background-color: #2BBBAD;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scrollbar-secondary::-webkit-scrollbar-track {
|
.scrollbar-secondary::-webkit-scrollbar-track {
|
||||||
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
||||||
background-color: #F5F5F5;
|
background-color: #F5F5F5;
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scrollbar-secondary::-webkit-scrollbar {
|
.scrollbar-secondary::-webkit-scrollbar {
|
||||||
width: 12px;
|
width: 12px;
|
||||||
background-color: #F5F5F5;
|
background-color: #F5F5F5;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scrollbar-secondary::-webkit-scrollbar-thumb {
|
.scrollbar-secondary::-webkit-scrollbar-thumb {
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
-webkit-box-shadow: inset 0 0 6px rgba(0, 0, 0, 0.1);
|
||||||
background-color: #aa66cc;
|
background-color: #aa66cc;
|
||||||
}
|
}
|
||||||
|
|
||||||
.fancytree-container {
|
.fancytree-container {
|
||||||
outline: none;
|
outline: none;
|
||||||
}
|
}
|
@ -3,7 +3,7 @@ import templates
|
|||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# SVG : Menu icon
|
# SVG : Menu icon
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
proc Menu_icon* (): string = tmpli html"""
|
proc Menu_icon*(): string = tmpli html"""
|
||||||
<svg class="menu-icon" version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg"
|
<svg class="menu-icon" version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg"
|
||||||
xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" viewBox="0 0 297 297"
|
xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" viewBox="0 0 297 297"
|
||||||
style="enable-background:new 0 0 297 297;" xml:space="preserve">
|
style="enable-background:new 0 0 297 297;" xml:space="preserve">
|
||||||
@ -119,7 +119,7 @@ proc Notebooks_icon*(): string = tmpli html"""
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# SVG : All tags icon
|
# SVG : Tags icon
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
proc Tags_icon*(): string = tmpli html"""
|
proc Tags_icon*(): string = tmpli html"""
|
||||||
<svg class="sidebar-icon" version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg"
|
<svg class="sidebar-icon" version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg"
|
||||||
@ -155,6 +155,26 @@ proc Tags_icon*(): string = tmpli html"""
|
|||||||
</svg>
|
</svg>
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
|
# SVG : Dashbord icon
|
||||||
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
|
proc Dashbord_icon*(): string = tmpli html"""
|
||||||
|
<svg class="svg-icon" viewBox="0 0 20 20">
|
||||||
|
<path fill="none" d="M7.228,11.464H1.996c-0.723,0-1.308,0.587-1.308,1.309v5.232c0,0.722,0.585,1.308,1.308,1.308h5.232
|
||||||
|
c0.723,0,1.308-0.586,1.308-1.308v-5.232C8.536,12.051,7.95,11.464,7.228,11.464z M7.228,17.351c0,0.361-0.293,0.654-0.654,0.654
|
||||||
|
H2.649c-0.361,0-0.654-0.293-0.654-0.654v-3.924c0-0.361,0.292-0.654,0.654-0.654h3.924c0.361,0,0.654,0.293,0.654,0.654V17.351z
|
||||||
|
M17.692,11.464H12.46c-0.723,0-1.308,0.587-1.308,1.309v5.232c0,0.722,0.585,1.308,1.308,1.308h5.232
|
||||||
|
c0.722,0,1.308-0.586,1.308-1.308v-5.232C19,12.051,18.414,11.464,17.692,11.464z M17.692,17.351c0,0.361-0.293,0.654-0.654,0.654
|
||||||
|
h-3.924c-0.361,0-0.654-0.293-0.654-0.654v-3.924c0-0.361,0.293-0.654,0.654-0.654h3.924c0.361,0,0.654,0.293,0.654,0.654V17.351z
|
||||||
|
M7.228,1H1.996C1.273,1,0.688,1.585,0.688,2.308V7.54c0,0.723,0.585,1.308,1.308,1.308h5.232c0.723,0,1.308-0.585,1.308-1.308
|
||||||
|
V2.308C8.536,1.585,7.95,1,7.228,1z M7.228,6.886c0,0.361-0.293,0.654-0.654,0.654H2.649c-0.361,0-0.654-0.292-0.654-0.654V2.962
|
||||||
|
c0-0.361,0.292-0.654,0.654-0.654h3.924c0.361,0,0.654,0.292,0.654,0.654V6.886z M17.692,1H12.46c-0.723,0-1.308,0.585-1.308,1.308
|
||||||
|
V7.54c0,0.723,0.585,1.308,1.308,1.308h5.232C18.414,8.848,19,8.263,19,7.54V2.308C19,1.585,18.414,1,17.692,1z M17.692,6.886
|
||||||
|
c0,0.361-0.293,0.654-0.654,0.654h-3.924c-0.361,0-0.654-0.292-0.654-0.654V2.962c0-0.361,0.293-0.654,0.654-0.654h3.924
|
||||||
|
c0.361,0,0.654,0.292,0.654,0.654V6.886z"></path>
|
||||||
|
</svg>
|
||||||
|
"""
|
||||||
|
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
# SVG : Settings icon
|
# SVG : Settings icon
|
||||||
# --==--==--==--==--==--==--==--==--==--==-- #
|
# --==--==--==--==--==--==--==--==--==--==-- #
|
||||||
@ -191,4 +211,4 @@ proc SendFeedback_icon*(): string = tmpli html"""
|
|||||||
</path>
|
</path>
|
||||||
</g>
|
</g>
|
||||||
</svg>
|
</svg>
|
||||||
"""
|
"""
|
||||||
|
@ -13,7 +13,7 @@
|
|||||||
#
|
#
|
||||||
# import snippet_html
|
# import snippet_html
|
||||||
# import snippet_icons
|
# import snippet_icons
|
||||||
#proc genSecret(c: var TData, columnLeftInfo: var ColomnLeftData): string =
|
#proc genSecret(c: var TData, columnLeftInfo: var ColomnLeftData, ColomnRightInfo: var ColomnRightData): string =
|
||||||
# result = ""
|
# result = ""
|
||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html lang="en">
|
<html lang="en">
|
||||||
@ -22,7 +22,7 @@
|
|||||||
<meta charset="UTF-8">
|
<meta charset="UTF-8">
|
||||||
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
<title>Document</title>
|
<title>Joplin-TheNewWeb</title>
|
||||||
<link rel="stylesheet" href="css/styles.css">
|
<link rel="stylesheet" href="css/styles.css">
|
||||||
<link rel="stylesheet" href="css/notif-bell.css">
|
<link rel="stylesheet" href="css/notif-bell.css">
|
||||||
<script src="js/script.js" defer></script>
|
<script src="js/script.js" defer></script>
|
||||||
@ -63,6 +63,7 @@
|
|||||||
# var active_notes: string = ""
|
# var active_notes: string = ""
|
||||||
# var active_notebooks: string = ""
|
# var active_notebooks: string = ""
|
||||||
# var active_tags: string = ""
|
# var active_tags: string = ""
|
||||||
|
# var active_dashbord: string = ""
|
||||||
# var active_settings: string = ""
|
# var active_settings: string = ""
|
||||||
# var active_sendFeedBack: string = ""
|
# var active_sendFeedBack: string = ""
|
||||||
# if columnLeftInfo.option == newNote:
|
# if columnLeftInfo.option == newNote:
|
||||||
@ -77,6 +78,8 @@
|
|||||||
# active_notebooks = "active"
|
# active_notebooks = "active"
|
||||||
# elif columnLeftInfo.option == tags:
|
# elif columnLeftInfo.option == tags:
|
||||||
# active_tags = "active"
|
# active_tags = "active"
|
||||||
|
# elif columnLeftInfo.option == dashbord:
|
||||||
|
# active_dashbord = "active"
|
||||||
# end if
|
# end if
|
||||||
|
|
||||||
<li class="sidebar-list-item ${active_newNote}">
|
<li class="sidebar-list-item ${active_newNote}">
|
||||||
@ -145,6 +148,18 @@
|
|||||||
<div class="hidden-sidebar">Tags</div>
|
<div class="hidden-sidebar">Tags</div>
|
||||||
</a>
|
</a>
|
||||||
</li>
|
</li>
|
||||||
|
|
||||||
|
<!--Dashbord dans la bar de navigation -->
|
||||||
|
<li class="sidebar-list-item ${active_dashbord}">
|
||||||
|
<a href="/secret?msg=dashbord" class="sidebar-link">
|
||||||
|
|
||||||
|
# var icon_Dashbord = Dashbord_icon()
|
||||||
|
${icon_Dashbord}
|
||||||
|
|
||||||
|
<div class="hidden-sidebar">Dashbord</div>
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
<div class="bottom-sidebar">
|
<div class="bottom-sidebar">
|
||||||
@ -225,14 +240,15 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
# elif columnLeftInfo.option == notes:
|
# elif columnLeftInfo.option == notes:
|
||||||
<h2>Notes</h2>
|
<h2>Notes</h2> <h5>${columnLeftInfo.j_notes_nb}</h5>
|
||||||
<div class="scrollbar scrollbar-primary">
|
<div class="scrollbar scrollbar-primary">
|
||||||
<div class="force-overflow">
|
<div class="force-overflow">
|
||||||
<div class="ColomnLeftJoplinNotes">
|
# for i in 0 .. (columnLeftInfo.j_notes.id.len() - 1) :
|
||||||
# for i in columnLeftInfo.j_notes.title:
|
<div class="columnLeftJoplinNotes">
|
||||||
<p> ${i} </p>
|
<a href="/secret?msg=notes¬eid=${columnLeftInfo.j_notes.id[i]}" class="sidebar-note-link">${columnLeftInfo.j_notes.title[i]}</a>
|
||||||
# end for
|
<p class="sidebar-note-date">${columnLeftInfo.j_notes.created_time[i]} </p>
|
||||||
</div>
|
</div>
|
||||||
|
# end for
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
# elif columnLeftInfo.option == notebooks:
|
# elif columnLeftInfo.option == notebooks:
|
||||||
@ -254,14 +270,37 @@
|
|||||||
# for i in columnLeftInfo.j_tags.title:
|
# for i in columnLeftInfo.j_tags.title:
|
||||||
<p> ${i} </p>
|
<p> ${i} </p>
|
||||||
# end for
|
# end for
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
# elif columnLeftInfo.option == dashbord:
|
||||||
|
<h2>Dashboard</h2>
|
||||||
|
<div class="scrollbar scrollbar-primary">
|
||||||
|
<div class="force-overflow">
|
||||||
|
<div class="ColomnLeftJoplinDashboard">
|
||||||
|
|
||||||
|
<p>DASHBORD</p>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
# end if
|
# end if
|
||||||
</div>
|
</div>
|
||||||
<div class="column right" style="background-color:#bbb;">
|
<div class="column right" >
|
||||||
<h2>Column 2</h2>
|
# if columnLeftInfo.option == newNote:
|
||||||
<p>Some text..</p>
|
<h2>New Note !!</h2>
|
||||||
|
<p>Some text..</p>
|
||||||
|
# elif columnLeftInfo.option == notes:
|
||||||
|
# if ColomnRightInfo.j_SelectedNote.id != "":
|
||||||
|
<h2> ${ColomnRightInfo.j_SelectedNote.title}</h2>
|
||||||
|
<hr width="65%"></hr>
|
||||||
|
<br>
|
||||||
|
<p>${ColomnRightInfo.j_SelectedNote.body}</p>
|
||||||
|
# else :
|
||||||
|
<h2>Click on note to see it !!</h2>
|
||||||
|
# end if
|
||||||
|
|
||||||
|
# end if
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
Loading…
Reference in New Issue
Block a user