2013-02-16 19:18:14 +01:00
<!DOCTYPE html>
2013-10-07 00:12:44 +02:00
< html ng-app = "weechat" ng-cloak >
2013-02-16 18:49:20 +01:00
< head >
2013-10-09 17:53:25 +02:00
< meta charset = "utf-8" >
< meta http-equiv = "X-UA-Compatible" content = "IE=Edge" >
2013-12-14 23:35:10 +01:00
< meta name = "viewport" content = "width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no" >
2014-01-25 14:55:08 +01:00
< meta name = "apple-mobile-web-app-capable" content = "yes" >
< meta name = "mobile-web-app-capable" content = "yes" >
2014-02-21 11:22:27 +01:00
< title ng-bind-template = "{{ notificationStatus }}WeeChat {{ pageTitle}}" > < / title >
2014-02-26 01:50:24 +01:00
< link href = "//netdna.bootstrapcdn.com/bootstrap/3.1.1/css/bootstrap.min.css" rel = "stylesheet" media = "screen" >
2014-03-04 04:13:41 +01:00
< link rel = "shortcut icon" sizes = "128x128" href = "assets/img/glowing_bear_128x128.png" >
< link rel = "apple-touch-icon" sizes = "128x128" href = "assets/img/glowing_bear_128x128.png" >
< link rel = "shortcut icon" type = "image/png" href = "assets/img/favicon.png" >
2013-10-26 10:30:35 +02:00
< link href = "css/style.css" rel = "stylesheet" media = "screen" >
2013-10-02 01:39:24 +02:00
< link href = "css/glowingbear.css" rel = "stylesheet" media = "screen" >
2014-02-16 21:33:13 +01:00
< script src = "//ajax.googleapis.com/ajax/libs/jquery/1.11.0/jquery.min.js" > < / script >
2014-04-26 22:56:30 +02:00
< script src = "//ajax.googleapis.com/ajax/libs/angularjs/1.2.16/angular.min.js" > < / script >
< script src = "//ajax.googleapis.com/ajax/libs/angularjs/1.2.16/angular-route.min.js" > < / script >
< script src = "//ajax.googleapis.com/ajax/libs/angularjs/1.2.16/angular-sanitize.min.js" > < / script >
< script src = "//ajax.googleapis.com/ajax/libs/angularjs/1.2.16/angular-touch.js" > < / script >
< script src = "//ajax.googleapis.com/ajax/libs/angularjs/1.2.16/angular-animate.js" > < / script >
< script src = "//cdnjs.cloudflare.com/ajax/libs/underscore.js/1.6.0/underscore-min.js" > < / script >
2013-12-20 14:09:08 +01:00
< script type = "text/javascript" src = "3rdparty/inflate.min.js" > < / script >
2013-10-05 16:05:16 +02:00
< script type = "text/javascript" src = "js/localstorage.js" > < / script >
2013-10-16 23:10:49 +02:00
< script type = "text/javascript" src = "js/weechat.js" > < / script >
2013-10-27 00:44:48 +02:00
< script type = "text/javascript" src = "js/irc-utils.js" > < / script >
2014-02-07 02:32:25 +01:00
< script type = "text/javascript" src = "js/glowingbear.js" > < / script >
2014-02-08 16:46:58 +01:00
< script type = "text/javascript" src = "js/websockets.js" > < / script >
2013-10-08 02:42:19 +02:00
< script type = "text/javascript" src = "js/models.js" > < / script >
2013-10-08 23:06:47 +02:00
< script type = "text/javascript" src = "js/plugins.js" > < / script >
2014-01-26 16:23:56 +01:00
< script type = "text/javascript" src = "3rdparty/bindonce.min.js" > < / script >
2014-02-02 13:55:28 +01:00
< script type = "text/javascript" src = "3rdparty/favico-0.3.4-mod.min.js" > < / script >
2014-02-26 01:50:24 +01:00
< script src = "//netdna.bootstrapcdn.com/bootstrap/3.1.1/js/bootstrap.min.js" > < / script >
2013-02-16 18:49:20 +01:00
< / head >
2013-10-11 16:24:25 +02:00
< body ng-controller = "WeechatCtrl" ng-keydown = "handleKeyPress($event)" >
2014-04-25 05:20:56 +02:00
< div ng-hide = "connected" class = "container" >
2013-10-09 17:53:25 +02:00
< h2 >
2014-03-09 12:22:52 +01:00
< img alt = "logo" src = "assets/img/glowing-bear.svg" >
2014-03-09 11:32:35 +01:00
< span > glowing bear< / span >
< small > WeeChat web frontend< / small >
2013-10-09 17:53:25 +02:00
< / h2 >
2014-05-09 18:49:07 +02:00
< div class = "alert alert-danger" ng-show = "errorMessage" >
< strong > Connection error< / strong > The client was unable to connect to the WeeChat relay
< / div >
< div class = "alert alert-danger" ng-show = "sslError" >
< strong > Secure connection error< / strong > A secure connection with the WeeChat relay could not be initiated. This is most likely because your browser does not trust your relay's certificate. Please read the encryption instructions below!
< / div >
< div class = "alert alert-danger" ng-show = "securityError" >
< strong > Secure connection error< / strong > Unable to connect to unencrypted relay when your are connecting to Glowing Bear over HTTPS. Please use an encrypted relay or load the page without using HTTPS.
< / div >
< div class = "panel-group" id = "accordion" >
< div class = "panel" >
< div class = "panel-heading" >
< h4 class = "panel-title" >
< a class = "accordion-toggle" data-toggle = "collapse" data-parent = "#accordion" href = "#collapseOne" >
Connection settings
< / a >
< / h4 >
< / div >
< div id = "collapseOne" class = "panel-collapse collapse in" >
< div class = "panel-body" >
< form class = "form-signin" role = "form" >
< div class = "form-group" >
< label class = "control-label" for = "host" > WeeChat relay hostname and port number< / label >
< div class = "input-group" >
< input type = "text" class = "form-control monospace" id = "host" ng-model = "host" placeholder = "Address" >
< input type = "text" class = "form-control monospace" id = "port" ng-model = "port" placeholder = "9001" >
< / div >
< label class = "control-label" for = "password" > WeeChat relay password< / label >
< input type = "password" class = "form-control monospace" id = "password" ng-model = "password" placeholder = "Password" >
< div class = "alert alert-danger" ng-show = "passwordError" >
Error: wrong password
< / div >
2014-02-14 15:14:55 +01:00
2014-05-09 18:49:07 +02:00
< div class = "checkbox" >
< label class = "control-label" for = "savepassword" >
< input type = "checkbox" id = "savepassword" ng-model = "savepassword" >
Save password in your browser
< / label >
2013-10-15 16:09:08 +02:00
< / div >
2014-05-09 18:49:07 +02:00
< div class = "checkbox" >
< label class = "control-label" for = "ssl" >
< input type = "checkbox" id = "ssl" ng-model = "ssl" >
Encryption. Read instructions for help
< / label >
< / div >
< / div >
< button class = "btn btn-lg btn-primary" ng-click = "connect()" > {{ connectbutton }} < i class = "glyphicon glyphicon-chevron-right" > < / i > < / button >
< / form >
2013-10-15 16:09:08 +02:00
< / div >
< / div >
2014-05-09 18:49:07 +02:00
< / div >
< div class = "panel" >
< div class = "panel-heading" >
< h4 class = "panel-title" >
< a class = "accordion-toggle" data-toggle = "collapse" data-parent = "#accordion" href = "#collapseTwo" >
Usage instructions
< / a >
< / h4 >
< / div >
< div id = "collapseTwo" class = "panel-collapse collapse" >
< div class = "panel-body" >
< h3 > Configuring the relay< / h3 >
< div > To start using glowing bear, please enable the relay plugin in your WeeChat client:
2013-10-26 08:36:06 +02:00
< pre >
/set relay.network.password yourpassword
/relay add weechat 9001
< / pre >
2014-05-09 18:49:07 +02:00
< span class = "label label-danger" > WeeChat version 0.4.2 or higher is required.< / span > < br >
The communication goes directly between your browser and your WeeChat relay in plain text. Check the instructions below for help on setting up encrypted communication.
Connection settings, including your password, are saved locally in your own browser between sessions.
< br >
< h3 > Shortcuts< / h3 >
Glowing Bear has a few shortcuts:
< ul >
< li > ALT-n: Toggle nicklist< / li >
< li > ALT-l: Focus on input bar< / li >
< li > ALT-[0-9]: Focus on buffer< / li >
< li > ALT-a: Focus on next buffer with activity< / li >
< li > ALT-< : Switch to previous buffer< / li >
< li > ALT-g: Focus on buffer list filter< / li >
< li > Esc-Esc: disconnect (double-tap)< / li >
< li > arrow keys: history navigation< / li >
< / ul >
2013-10-15 16:09:08 +02:00
< / div >
< / div >
< / div >
2014-05-09 18:49:07 +02:00
< / div >
< div class = "panel " >
< div class = "panel-heading" >
< h4 class = "panel-title" >
< a class = "accordion-toggle" data-toggle = "collapse" data-parent = "#accordion" href = "#collapseThree" >
Encryption instructions
< / a >
< / h4 >
< / div >
< div id = "collapseThree" class = "panel-collapse collapse" >
< div class = "panel-body" >
< p > If you check the encryption box, the communication between browser and WeeChat will be encrypted with SSL.< / p >
< p > < strong > Note< / strong > : If you are using a self-signed certificate, you have to visit < a href = "https://{{ host }}:{{ port }}/" > https://{{ host || 'weechathost' }}:{{ port || 'relayport' }}/< / a > in your browser first to add a security exception. You can close that tab once you confirmed the certificate, no content will appear. The necessity of this process is a bug in < a href = "https://bugzilla.mozilla.org/show_bug.cgi?id=594502" > Firefox< / a > and other browsers.< / p >
< p > < strong > Setup< / strong > : If you want to use an encrypted session you first have to set up the relay to use SSL. You basically have two options: a self-signed certificate is easier to set up, but requires manual security exceptions. Using a certificate that is trusted by your browser requires more setup, but does not require any security exceptions. As the process for requesting a certificate is different for every certification authority, we detail the method for setting up WeeChat with a self-signed certificate here. To create one, execute the following commands in a shell on the same host and as the user running WeeChat:< / p >
2013-10-26 08:36:06 +02:00
< pre >
$ mkdir -p ~/.weechat/ssl
$ cd ~/.weechat/ssl
2014-05-04 20:07:08 +02:00
$ openssl req -nodes -newkey rsa:4096 -keyout relay.pem -x509 -days 365 -out relay.pem -subj "/CN={{host || 'your weechat host'}}/"
2013-10-26 08:36:06 +02:00
< / pre >
2014-05-09 18:49:07 +02:00
< p > If WeeChat is already running, you can reload the certificate and private key and set up an encrypted relay on port 8000 with these WeeChat commands:< / p >
2013-10-26 08:36:06 +02:00
< pre >
/relay sslcertkey
/relay add ssl.weechat 8000
< / pre >
2013-10-21 15:51:05 +02:00
< / div >
< / div >
2014-05-09 18:49:07 +02:00
< / div >
< div class = "panel" ng-hide = "isinstalled" >
< div class = "panel-heading" >
< h4 class = "panel-title" >
< a class = "accordion-toggle" data-toggle = "collapse" data-parent = "#accordion" href = "#collapseFour" >
Install app
< / a >
< / h4 >
2013-10-15 16:09:08 +02:00
< / div >
2014-05-09 18:49:07 +02:00
< div id = "collapseFour" class = "panel-collapse collapse" >
< div class = "panel-body" >
< p > You don't need to install anything to use this app, it should work with any modern browser. Start using it < a data-toggle = "collapse" data-parent = "#accordion" href = "#collapseOne" > right now< / a > ! However, there are a few ways to improve integration with your operating system.< / p >
< h3 > Firefox< / h3 >
< p > If you have a recent version of Firefox you can install glowing bear as an app. Click the button to install.< / p >
< button class = "btn btn-lg btn-primary" ng-click = "install()" > Install Firefox app < i class = "glyphicon glyphicon-chevron-right" > < / i > < / button >
< h3 > Chrome< / h3 >
< p > To install glowing bear as an app in Chrome, select < kbd > Menu - Add to home screen< / kbd > (Android) or < kbd > Menu - Tools - Create Application Shortcuts< / kbd > (desktop version).< / p >
2014-02-13 00:03:22 +01:00
< / div >
2014-05-09 18:49:07 +02:00
< / div >
< / div >
< div class = "panel" >
< div class = "panel-heading" >
< h4 class = "panel-title" >
< a class = "accordion-toggle" data-toggle = "collapse" data-parent = "#accordion" href = "#collapseFive" >
Get involved
< / a >
< / h4 >
< / div >
< div id = "collapseFive" class = "panel-collapse collapse" >
< div class = "panel-body" >
< p > Glowing bear is built by a small group of developers in their free time. As we're always trying to improve it, we would love getting your feedback and help. If that sounds like something you might enjoy, check out our < a href = "https://github.com/glowing-bear/glowing-bear" > project page< / a > on GitHub!< / p >
< p > If you're interested in contributing or simply want to say hello, head over to < strong > #glowing-bear< / strong > on < strong > freenode!< / strong > We won't bite, promise (-ish).< / p >
2014-02-13 00:03:22 +01:00
< / div >
< / div >
2013-10-11 23:21:51 +02:00
< / div >
2013-10-11 23:38:30 +02:00
< / div >
< / div >
2013-10-09 17:53:25 +02:00
< div class = "content" ng-show = "connected" >
2014-02-13 00:04:24 +01:00
< div id = "topbar" >
< div class = "brand" >
2014-02-21 09:32:52 +01:00
< a href = "#" ng-click = "swipeSidebar()" >
2014-03-04 04:13:41 +01:00
< img alt = "brand" src = "assets/img/favicon.png" title = "Connected to {{ host }}:{{ port}}" >
2014-02-13 00:04:24 +01:00
< / a >
2014-04-26 17:49:56 +02:00
< button ng-if = "debugMode" ng-click = "countWatchers()" > Count< br / > Watchers< / button >
2014-02-13 00:04:24 +01:00
< / div >
2014-02-16 15:10:56 +01:00
< div class = "title" ng-bind-html = "activeBuffer().title | irclinky:'_blank'" > < / div >
2014-02-13 00:04:24 +01:00
< div class = "actions pull-right vertical-line-left" >
2014-02-26 02:09:10 +01:00
< div class = "pull-left" >
< a class = "" data-toggle = "modal" data-target = "#settingsModal" title = "Options menu" >
2014-02-13 00:04:24 +01:00
< i class = "glyphicon glyphicon-cog" > < / i >
< / a >
2014-02-26 02:09:10 +01:00
< / div >
< a ng-click = "disconnect()" title = "Disconnect from WeeChat" >
< i class = "glyphicon glyphicon-remove" > < / i >
< / a >
< / div >
< / div >
2014-04-25 05:20:56 +02:00
< div bindonce id = "sidebar" ng-show = "showSidebar" ng-swipe-left = "swipeSidebar()" class = "vertical-line" >
2014-02-26 02:09:10 +01:00
< ul class = "nav nav-pills nav-stacked" ng-class = "{'indented': (predicate === 'serverSortKey')}" >
< li class = "bufferfilter" >
< form role = "form" >
< input class = "form-control monospace" type = "text" id = "bufferFilter" ng-model = "search" ng-keydown = "handleSearchBoxKey($event)" placeholder = "Search" >
< / form >
< / li >
2014-02-28 03:53:30 +01:00
< li class = "buffer" ng-class = "{'active': buffer.active, 'indent': buffer.indent }" ng-repeat = "(key, buffer) in (filteredBuffers = (buffers | toArray | filter:{fullName:search} | filter:hasUnread | orderBy:predicate))" >
< a href = "#" ng-click = "setActiveBuffer(buffer.id)" title = "{{ buffer.fullName }}" >
2014-04-26 17:49:56 +02:00
< span class = "badge pull-right" ng-if = "buffer.unread && !buffer.notification" ng-bind = "buffer.unread" > < / span >
< span class = "badge pull-right danger" ng-if = "buffer.notification" ng-bind = "buffer.notification" > < / span >
< span class = "buffername" > {{ buffer.shortName }}< / span > < span ng-if = "!buffer.shortName" > {{ buffer.fullName }}< / span >
2014-02-26 02:09:10 +01:00
< / a >
< / li >
< / ul >
< / div >
2014-02-28 00:00:37 +01:00
< div bindonce id = "bufferlines" class = "monospace" ng-swipe-right = "swipeSidebar()" ng-swipe-left = "openNick()" ng-class = "{'withnicklist': showNicklist, 'withsidebar': showSidebar}" >
2014-04-26 17:49:56 +02:00
< div id = "nicklist" ng-if = "showNicklist" ng-swipe-right = "closeNick()" class = "vertical-line-left" >
2014-04-20 17:44:50 +02:00
< ul class = "nicklistgroup list-unstyled" ng-repeat = "group in nicklist" >
2014-02-26 02:09:10 +01:00
< li ng-repeat = "nick in group.nicks|orderBy:'name'" ng-click = "openBuffer(nick.name)" >
< a ng-click = "nickAction(nick)" > < span bo-class = "nick.prefixClasses" bo-text = "nick.prefix" > < / span > < span bo-class = "nick.nameClasses" bo-text = "nick.name" > < / span > < / a >
< / li >
< / ul >
< / div >
2014-02-28 00:00:37 +01:00
< table ng-class = "{'notimestamp':notimestamp}" >
2014-02-26 02:09:10 +01:00
< tbody >
< tr class = "bufferline" >
2014-02-28 03:50:50 +01:00
< td ng-hide = "activeBuffer().allLinesFetched" colspan = "3" >
2014-02-26 02:09:10 +01:00
< a class = "fetchmorelines" ng-click = "fetchMoreLines()" ng-hide = "loadingLines" > Fetch more lines< / a >
< span ng-show = "loadingLines" > Fetching more lines...< / span >
< / td >
< / tr >
< / tbody >
2014-04-20 17:44:50 +02:00
< tbody ng-repeat = "bufferline in bufferlines" >
2014-02-26 02:09:10 +01:00
< tr class = "bufferline" >
2014-04-26 17:34:02 +02:00
< td class = "time" >
2014-02-26 02:09:10 +01:00
< span class = "date" bo-class = "{'repeated-time': bufferline.shortTime==bufferlines[$index-1].shortTime}" >
< span class = "cof-chat_time cob-chat_time coa-chat_time" bo-text = "bufferline.date|date:'HH'" > < / span > < span class = "cof-chat_time_delimiters cob-chat_time_delimiters coa-chat_time_delimiters" > :< / span > < span class = "cof-chat_time cob-chat_time coa-chat_time" bo-text = "bufferline.date|date:'mm'" > < / span >
< / span >
< / td >
2014-05-09 10:13:13 +02:00
< td class = "prefix" > < a ng-click = "addMention(bufferline.prefix)" > < span ng-repeat = "part in bufferline.prefix" bo-class = "part.classes" bo-html = "part.text" > < / span > < / a > < / td >
2014-02-26 02:09:10 +01:00
< td class = "message" >
< div ng-repeat = "metadata in bufferline.metadata" >
< div plugin data = "metadata" > < / div >
< / div >
< span ng-repeat = "part in bufferline.content" class = "text" bo-class = "part.classes" bo-html = "part.text|irclinky:'_blank'" > < / span >
< / td >
< / tr >
< tr class = "readmarker" ng-if = "activeBuffer().lastSeen==$index" >
< td colspan = "3" >
< hr id = "readmarker" >
< / td >
< / tr >
< / tbody >
< / table >
< / div >
2014-04-29 01:13:41 +02:00
< div class = "footer" ng-class = "{'withnicklist': showNicklist, 'withsidebar': showSidebar}" >
2014-02-26 02:09:10 +01:00
< div input-bar input-id = "sendMessage" > < / div >
2014-03-01 19:55:22 +01:00
< / div >
2014-04-29 01:13:41 +02:00
< / div >
2014-05-09 18:49:07 +02:00
< div id = "soundNotification" > < / div >
2014-02-26 02:09:10 +01:00
< div id = "settingsModal" class = "modal fade" >
< div class = "modal-dialog" >
< div class = "modal-content" >
< div class = "modal-header" >
< button type = "button" class = "close" data-dismiss = "modal" aria-hidden = "true" > × < / button >
< h4 class = "modal-title" > Settings< / h4 >
< / div >
< div class = "modal-body" >
< p > Settings will be stored in your browser.< / p >
< ul class = "" >
2014-02-13 00:04:24 +01:00
< li >
< form class = "form-inline" role = "form" >
< div class = "checkbox" >
< label >
< input type = "checkbox" ng-model = "onlyUnread" >
Only show unread
< / label >
< / div >
< / form >
< / li >
< li >
< form class = "form-inline" role = "form" >
< div class = "checkbox" >
2014-02-16 21:38:10 +01:00
< label >
2014-02-13 00:04:24 +01:00
< input type = "checkbox" ng-model = "notimestamp" >
Hide timestamps
< / label >
< / div >
< / form >
< / li >
< li >
< form class = "form-inline" role = "form" >
< div class = "checkbox" >
< label >
< input type = "checkbox" ng-model = "noembed" >
Hide embedded content by default
< / label >
< / div >
< / form >
< / li >
< li >
< form class = "form-inline" role = "form" >
< div class = "checkbox" >
< label >
< input type = "checkbox" ng-model = "hotlistsync" >
Sync hotlist with WeeChat
< / label >
< / div >
< / form >
< / li >
< li >
< form class = "form-inline" role = "form" >
< div class = "checkbox" >
< label >
< input type = "checkbox" ng-model = "nonicklist" >
Hide nicklist
< / label >
< / div >
< / form >
< / li >
< li >
< form class = "form-inline" role = "form" >
< div class = "checkbox" >
< label >
< input type = "checkbox" ng-model = "orderbyserver" >
Order channels by server
< / label >
< / div >
< / form >
< / li >
2014-02-20 23:35:34 +01:00
< li >
< form class = "form-inline" role = "form" >
< div class = "checkbox" >
< label >
< input type = "checkbox" ng-model = "useFavico" >
Display unread count in favicon
< / label >
< / div >
< / form >
< / li >
2014-03-01 19:55:22 +01:00
< li >
< form class = "form-inline" role = "form" >
< div class = "checkbox" >
< label >
< input type = "checkbox" ng-model = "soundnotification" >
Play sound on notification
< / label >
< / div >
< / form >
< / li >
2014-02-13 00:04:24 +01:00
< li >
< form class = "form-inline" role = "form" >
< div class = "checkbox" >
< label >
< input type = "checkbox" ng-model = "debugMode" >
Debug Mode
< / label >
< / div >
< / form >
< / li >
< / ul >
< / div >
2014-02-26 02:09:10 +01:00
< div class = "modal-footer" >
< button type = "button" class = "btn btn-primary" data-dismiss = "modal" > Close< / button >
< / div >
< / div > <!-- /.modal - content -->
< / div > <!-- /.modal - dialog -->
< / div > <!-- /.modal -->
2013-10-05 18:40:01 +02:00
< / body >
2013-02-16 18:49:20 +01:00
< / html >