mirror of
https://github.com/ilyakooo0/urbit.git
synced 2024-12-14 17:41:33 +03:00
Merge branch 'master' into release/next-vere
This commit is contained in:
commit
e40bd3cf6f
@ -175,14 +175,15 @@ the pill to have the new files/hash. For most things, it is sufficient to run
|
|||||||
However, if you've made a change to Landscape's JS, then you will need to build
|
However, if you've made a change to Landscape's JS, then you will need to build
|
||||||
a "glob" and upload it to bootstrap.urbit.org. To do this, run `npm install;
|
a "glob" and upload it to bootstrap.urbit.org. To do this, run `npm install;
|
||||||
npm run build:prod` in `pkg/interface`, and add the resulting
|
npm run build:prod` in `pkg/interface`, and add the resulting
|
||||||
`pkg/arvo/app/landscape/index.js` to a fakezod at that path (or just create a
|
`pkg/arvo/app/landscape/index.[hash].js` to a fakezod at that path (or just create a
|
||||||
new fakezod with `urbit -F zod -B bin/solid.pill -A pkg/arvo`). Run
|
new fakezod with `urbit -F zod -B bin/solid.pill -A pkg/arvo`). Run
|
||||||
`:glob|make`, and this will output a file in `fakezod/.urb/put/glob-0vXXX.glob`.
|
`:glob|make`, and this will output a file in `fakezod/.urb/put/glob-0vXXX.glob`.
|
||||||
|
|
||||||
Upload this file to bootstrap.urbit.org, and modify `+hash` at the top of
|
Upload this file to bootstrap.urbit.org, and modify `+hash` at the top of
|
||||||
`pkg/arvo/app/glob.hoon` to match the hash in the filename. Do not commit the
|
`pkg/arvo/app/glob.hoon` to match the hash in the filename of the `.glob` file.
|
||||||
produced `index.js` and make sure it doesn't end up in your pills (they should
|
Amend `pkg/arvo/app/landscape/index.html` to import the hashed JS bundle, instead
|
||||||
be less than 10MB each).
|
of the unversioned index.js. Do not commit the produced `index.js` and
|
||||||
|
make sure it doesn't end up in your pills (they should be less than 10MB each).
|
||||||
|
|
||||||
### Tag the resulting commit
|
### Tag the resulting commit
|
||||||
|
|
||||||
|
@ -1,3 +1,3 @@
|
|||||||
version https://git-lfs.github.com/spec/v1
|
version https://git-lfs.github.com/spec/v1
|
||||||
oid sha256:d7088528dbfd54a913921ade093251d678c4ccebfd0ad85ef2022520266b3954
|
oid sha256:6cd7246753c12c7acb757e1a6ee54c177806c20a137ad8fb4300c000ac146a0f
|
||||||
size 16451173
|
size 6260139
|
||||||
|
@ -268,7 +268,7 @@
|
|||||||
%group-store
|
%group-store
|
||||||
%group-push-hook
|
%group-push-hook
|
||||||
=/ =cage
|
=/ =cage
|
||||||
:- %group-action
|
:- %group-update
|
||||||
!> ^- action:group-store
|
!> ^- action:group-store
|
||||||
[%change-policy rid %invite %add-invites (sy ship ~)]
|
[%change-policy rid %invite %add-invites (sy ship ~)]
|
||||||
[%pass / %agent [entity.rid app] %poke cage]
|
[%pass / %agent [entity.rid app] %poke cage]
|
||||||
|
@ -7,17 +7,20 @@
|
|||||||
$% [%clay =path]
|
$% [%clay =path]
|
||||||
[%glob =glob:glob]
|
[%glob =glob:glob]
|
||||||
==
|
==
|
||||||
+$ state-1
|
+$ state-base
|
||||||
$: %1
|
$: =configuration:srv
|
||||||
=configuration:srv
|
|
||||||
=serving
|
=serving
|
||||||
==
|
==
|
||||||
|
+$ state-2
|
||||||
|
$: %2
|
||||||
|
state-base
|
||||||
|
==
|
||||||
--
|
--
|
||||||
::
|
::
|
||||||
%+ verb |
|
%+ verb |
|
||||||
%- agent:dbug
|
%- agent:dbug
|
||||||
::
|
::
|
||||||
=| state-1
|
=| state-2
|
||||||
=* state -
|
=* state -
|
||||||
^- agent:gall
|
^- agent:gall
|
||||||
|_ =bowl:gall
|
|_ =bowl:gall
|
||||||
@ -60,12 +63,18 @@
|
|||||||
^- [content ?]
|
^- [content ?]
|
||||||
[[%clay clay-path] public]
|
[[%clay clay-path] public]
|
||||||
==
|
==
|
||||||
?> ?=(%1 -.old-state)
|
=? old-state ?=(%1 -.old-state)
|
||||||
|
%= old-state
|
||||||
|
- %2
|
||||||
|
serving (~(del by serving.old-state) /'~landscape'/js/index)
|
||||||
|
==
|
||||||
|
?> ?=(%2 -.old-state)
|
||||||
[~ this(state old-state)]
|
[~ this(state old-state)]
|
||||||
::
|
::
|
||||||
+$ versioned-state
|
+$ versioned-state
|
||||||
$% state-1
|
$% state-0
|
||||||
state-0
|
state-1
|
||||||
|
state-2
|
||||||
==
|
==
|
||||||
::
|
::
|
||||||
+$ serving-0 (map url-base=path [=clay=path public=?])
|
+$ serving-0 (map url-base=path [=clay=path public=?])
|
||||||
@ -74,6 +83,10 @@
|
|||||||
=configuration:srv
|
=configuration:srv
|
||||||
=serving-0
|
=serving-0
|
||||||
==
|
==
|
||||||
|
+$ state-1
|
||||||
|
$: %1
|
||||||
|
state-base
|
||||||
|
==
|
||||||
--
|
--
|
||||||
::
|
::
|
||||||
++ on-poke
|
++ on-poke
|
||||||
@ -169,7 +182,7 @@
|
|||||||
?~ content [not-found:gen %.n]
|
?~ content [not-found:gen %.n]
|
||||||
?- -.content.u.content
|
?- -.content.u.content
|
||||||
%clay
|
%clay
|
||||||
=/ scry-path
|
=/ scry-path=path
|
||||||
:* (scot %p our.bowl)
|
:* (scot %p our.bowl)
|
||||||
q.byk.bowl
|
q.byk.bowl
|
||||||
(scot %da now.bowl)
|
(scot %da now.bowl)
|
||||||
@ -179,10 +192,16 @@
|
|||||||
=/ file (as-octs:mimes:html .^(@ %cx scry-path))
|
=/ file (as-octs:mimes:html .^(@ %cx scry-path))
|
||||||
:_ public.u.content
|
:_ public.u.content
|
||||||
?+ ext.req-line not-found:gen
|
?+ ext.req-line not-found:gen
|
||||||
[~ %html] (html-response:gen file)
|
|
||||||
[~ %js] (js-response:gen file)
|
[~ %js] (js-response:gen file)
|
||||||
[~ %css] (css-response:gen file)
|
[~ %css] (css-response:gen file)
|
||||||
[~ %png] (png-response:gen file)
|
[~ %png] (png-response:gen file)
|
||||||
|
::
|
||||||
|
[~ %html]
|
||||||
|
%. file
|
||||||
|
%* . html-response:gen
|
||||||
|
cache
|
||||||
|
!=(/app/landscape/index/html (slag 3 scry-path))
|
||||||
|
==
|
||||||
==
|
==
|
||||||
::
|
::
|
||||||
%glob
|
%glob
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
/- glob
|
/- glob
|
||||||
/+ default-agent, verb, dbug
|
/+ default-agent, verb, dbug
|
||||||
|%
|
|%
|
||||||
++ hash 0v3.cus8h.vc64c.rfb3t.22oji.b529a
|
++ hash 0v2.pbthv.gd1q2.h2ura.5esrn.d361c
|
||||||
+$ state-0 [%0 hash=@uv glob=(unit (each glob:glob tid=@ta))]
|
+$ state-0 [%0 hash=@uv glob=(unit (each glob:glob tid=@ta))]
|
||||||
+$ all-states
|
+$ all-states
|
||||||
$% state-0
|
$% state-0
|
||||||
@ -41,7 +41,7 @@
|
|||||||
--
|
--
|
||||||
=| state=state-0
|
=| state=state-0
|
||||||
=. hash.state hash
|
=. hash.state hash
|
||||||
=/ serve-path=path /'~landscape'/js/index
|
=/ serve-path=path /'~landscape'/js/bundle
|
||||||
^- agent:gall
|
^- agent:gall
|
||||||
%+ verb |
|
%+ verb |
|
||||||
%- agent:dbug
|
%- agent:dbug
|
||||||
@ -82,9 +82,19 @@
|
|||||||
:_ this
|
:_ this
|
||||||
=/ home=path /(scot %p our.bowl)/home/(scot %da now.bowl)
|
=/ home=path /(scot %p our.bowl)/home/(scot %da now.bowl)
|
||||||
=+ .^(=tube:clay %cc (weld home /js/mime))
|
=+ .^(=tube:clay %cc (weld home /js/mime))
|
||||||
=+ .^(js=@t %cx (weld home /app/landscape/js/index/js))
|
=+ .^(arch %cy (weld home /app/landscape/js/bundle))
|
||||||
|
=/ bundle=path
|
||||||
|
%- need
|
||||||
|
^- (unit path)
|
||||||
|
%- ~(rep by dir)
|
||||||
|
|= [[file=@t ~] out=(unit path)]
|
||||||
|
?^ out out
|
||||||
|
?. =((end 3 5 file) 'index')
|
||||||
|
~
|
||||||
|
`/[file]/js
|
||||||
|
=+ .^(js=@t %cx :(weld home /app/landscape/js/bundle bundle))
|
||||||
=+ !<(=mime (tube !>(js)))
|
=+ !<(=mime (tube !>(js)))
|
||||||
=/ =glob:glob (~(put by *glob:glob) /js mime)
|
=/ =glob:glob (~(put by *glob:glob) bundle mime)
|
||||||
=/ =path /(cat 3 'glob-' (scot %uv (sham glob)))/glob
|
=/ =path /(cat 3 'glob-' (scot %uv (sham glob)))/glob
|
||||||
[%pass /make %agent [our.bowl %hood] %poke %drum-put !>([path (jam glob)])]~
|
[%pass /make %agent [our.bowl %hood] %poke %drum-put !>([path (jam glob)])]~
|
||||||
::
|
::
|
||||||
|
BIN
pkg/arvo/app/landscape/img/groups.png
Normal file
BIN
pkg/arvo/app/landscape/img/groups.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 693 B |
Binary file not shown.
Before Width: | Height: | Size: 255 B After Width: | Height: | Size: 582 B |
@ -4,7 +4,7 @@
|
|||||||
<title>OS1</title>
|
<title>OS1</title>
|
||||||
<meta charset="utf-8" />
|
<meta charset="utf-8" />
|
||||||
<meta name="viewport"
|
<meta name="viewport"
|
||||||
content="width=device-width, initial-scale=1, shrink-to-fit=no"/>
|
content="width=device-width, initial-scale=1, shrink-to-fit=no,maximum-scale=1"/>
|
||||||
<meta name="apple-mobile-web-app-capable" content="yes" />
|
<meta name="apple-mobile-web-app-capable" content="yes" />
|
||||||
<meta name="apple-touch-fullscreen" content="yes" />
|
<meta name="apple-touch-fullscreen" content="yes" />
|
||||||
<meta name="apple-mobile-web-app-status-bar-style" content="default" />
|
<meta name="apple-mobile-web-app-status-bar-style" content="default" />
|
||||||
@ -23,7 +23,7 @@
|
|||||||
<div id="root"/>
|
<div id="root"/>
|
||||||
<script src="/~landscape/js/channel.js"></script>
|
<script src="/~landscape/js/channel.js"></script>
|
||||||
<script src="/~landscape/js/session.js"></script>
|
<script src="/~landscape/js/session.js"></script>
|
||||||
<script src="/~landscape/js/index.js"></script>
|
<script src="/~landscape/js/bundle/index.f58fbbc4b037bb976a2a.js"></script>
|
||||||
<script src="https://sdk.amazonaws.com/js/aws-sdk-2.1.12.min.js"></script>
|
<script src="https://sdk.amazonaws.com/js/aws-sdk-2.1.12.min.js"></script>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
@ -1831,6 +1831,8 @@
|
|||||||
::
|
::
|
||||||
%subscribe
|
%subscribe
|
||||||
?> (team:title our.bol src.bol)
|
?> (team:title our.bol src.bol)
|
||||||
|
?: =(our.bol who.act)
|
||||||
|
[~ state]
|
||||||
=/ join-wire=wire
|
=/ join-wire=wire
|
||||||
/join-group/[(scot %p who.act)]/[book.act]
|
/join-group/[(scot %p who.act)]/[book.act]
|
||||||
=/ meta=(unit (set path))
|
=/ meta=(unit (set path))
|
||||||
|
@ -80,9 +80,11 @@
|
|||||||
++ max-1-wk ['cache-control' 'max-age=604800']
|
++ max-1-wk ['cache-control' 'max-age=604800']
|
||||||
::
|
::
|
||||||
++ html-response
|
++ html-response
|
||||||
|
=| cache=?
|
||||||
|= =octs
|
|= =octs
|
||||||
^- simple-payload:http
|
^- simple-payload:http
|
||||||
[[200 [['content-type' 'text/html'] max-1-wk ~]] `octs]
|
:_ `octs
|
||||||
|
[200 [['content-type' 'text/html'] ?:(cache [max-1-wk ~] ~)]]
|
||||||
::
|
::
|
||||||
++ js-response
|
++ js-response
|
||||||
|= =octs
|
|= =octs
|
||||||
|
186
pkg/interface/.eslintrc.js
Normal file
186
pkg/interface/.eslintrc.js
Normal file
@ -0,0 +1,186 @@
|
|||||||
|
const env = {
|
||||||
|
"browser": true,
|
||||||
|
"es6": true,
|
||||||
|
"node": true
|
||||||
|
};
|
||||||
|
|
||||||
|
const rules = {
|
||||||
|
"array-bracket-spacing": ["error", "never"],
|
||||||
|
"arrow-parens": [
|
||||||
|
"error",
|
||||||
|
"as-needed",
|
||||||
|
{
|
||||||
|
"requireForBlockBody": true
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"arrow-spacing": "error",
|
||||||
|
"block-spacing": ["error", "always"],
|
||||||
|
"brace-style": ["error", "1tbs"],
|
||||||
|
"camelcase": [
|
||||||
|
"error",
|
||||||
|
{
|
||||||
|
"properties": "never"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"comma-dangle": ["error", "never"],
|
||||||
|
"eol-last": ["error", "always"],
|
||||||
|
"func-name-matching": "error",
|
||||||
|
"indent": [
|
||||||
|
"off",
|
||||||
|
2,
|
||||||
|
{
|
||||||
|
"ArrayExpression": "off",
|
||||||
|
"SwitchCase": 1,
|
||||||
|
"CallExpression": {
|
||||||
|
"arguments": "off"
|
||||||
|
},
|
||||||
|
"FunctionDeclaration": {
|
||||||
|
"parameters": "off"
|
||||||
|
},
|
||||||
|
"FunctionExpression": {
|
||||||
|
"parameters": "off"
|
||||||
|
},
|
||||||
|
"MemberExpression": "off",
|
||||||
|
"ObjectExpression": "off",
|
||||||
|
"ImportDeclaration": "off"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"handle-callback-err": "off",
|
||||||
|
"linebreak-style": ["error", "unix"],
|
||||||
|
"max-lines": [
|
||||||
|
"error",
|
||||||
|
{
|
||||||
|
"max": 300,
|
||||||
|
"skipBlankLines": true,
|
||||||
|
"skipComments": true
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"max-lines-per-function": [
|
||||||
|
"warn",
|
||||||
|
{
|
||||||
|
"skipBlankLines": true,
|
||||||
|
"skipComments": true
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"max-statements-per-line": [
|
||||||
|
"error",
|
||||||
|
{
|
||||||
|
"max": 1
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"new-cap": [
|
||||||
|
"error",
|
||||||
|
{
|
||||||
|
"newIsCap": true,
|
||||||
|
"capIsNew": false
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"new-parens": "error",
|
||||||
|
"no-buffer-constructor": "error",
|
||||||
|
"no-console": "off",
|
||||||
|
"no-extra-semi": "off",
|
||||||
|
"no-fallthrough": "off",
|
||||||
|
"no-func-assign": "off",
|
||||||
|
"no-implicit-coercion": "error",
|
||||||
|
"no-multi-assign": "error",
|
||||||
|
"no-multiple-empty-lines": [
|
||||||
|
"error",
|
||||||
|
{
|
||||||
|
"max": 1
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"no-nested-ternary": "error",
|
||||||
|
"no-param-reassign": "off",
|
||||||
|
"no-return-assign": "error",
|
||||||
|
"no-return-await": "off",
|
||||||
|
"no-shadow-restricted-names": "error",
|
||||||
|
"no-tabs": "error",
|
||||||
|
"no-trailing-spaces": "error",
|
||||||
|
"no-unused-vars": [
|
||||||
|
"error",
|
||||||
|
{
|
||||||
|
"vars": "all",
|
||||||
|
"args": "none",
|
||||||
|
"ignoreRestSiblings": false
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"no-use-before-define": [
|
||||||
|
"error",
|
||||||
|
{
|
||||||
|
"functions": false,
|
||||||
|
"classes": false
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"no-useless-escape": "off",
|
||||||
|
"no-var": "error",
|
||||||
|
"nonblock-statement-body-position": ["error", "below"],
|
||||||
|
"object-curly-spacing": ["error", "always"],
|
||||||
|
"padded-blocks": ["error", "never"],
|
||||||
|
"prefer-arrow-callback": "error",
|
||||||
|
"prefer-const": [
|
||||||
|
"error",
|
||||||
|
{
|
||||||
|
"destructuring": "all",
|
||||||
|
"ignoreReadBeforeAssign": true
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"prefer-template": "off",
|
||||||
|
"quotes": ["error", "single"],
|
||||||
|
"semi": ["error", "always"],
|
||||||
|
"spaced-comment": [
|
||||||
|
"error",
|
||||||
|
"always",
|
||||||
|
{
|
||||||
|
"exceptions": ["!"]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"space-before-blocks": "error",
|
||||||
|
"unicode-bom": ["error", "never"],
|
||||||
|
"valid-jsdoc": "error",
|
||||||
|
"wrap-iife": ["error", "inside"],
|
||||||
|
"react/jsx-closing-bracket-location": 1,
|
||||||
|
"react/jsx-tag-spacing": 1,
|
||||||
|
"react/jsx-max-props-per-line": ["error", { "maximum": 2, "when": "multiline" }],
|
||||||
|
"react/prop-types": 0
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
"env": env,
|
||||||
|
"extends": [
|
||||||
|
"plugin:react/recommended",
|
||||||
|
"eslint:recommended",
|
||||||
|
],
|
||||||
|
"settings": {
|
||||||
|
"react": {
|
||||||
|
"version": "^16.5.2"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"parser": "babel-eslint",
|
||||||
|
"parserOptions": {
|
||||||
|
"ecmaVersion": 10,
|
||||||
|
"requireConfigFile": false,
|
||||||
|
"sourceType": "module"
|
||||||
|
},
|
||||||
|
"root": true,
|
||||||
|
"rules": rules,
|
||||||
|
"overrides": [
|
||||||
|
{
|
||||||
|
"files": ["**/*.ts", "**/*.tsx"],
|
||||||
|
"env": env,
|
||||||
|
"extends": [
|
||||||
|
"eslint:recommended",
|
||||||
|
"plugin:@typescript-eslint/eslint-recommended",
|
||||||
|
"plugin:@typescript-eslint/recommended"
|
||||||
|
],
|
||||||
|
"parser": "@typescript-eslint/parser",
|
||||||
|
"parserOptions": {
|
||||||
|
"ecmaFeatures": { "jsx": true },
|
||||||
|
"ecmaVersion": 10,
|
||||||
|
"requireConfigFile": false,
|
||||||
|
"sourceType": "module"
|
||||||
|
},
|
||||||
|
"plugins": ["@typescript-eslint"],
|
||||||
|
"rules": rules
|
||||||
|
}
|
||||||
|
]
|
||||||
|
};
|
@ -1,147 +0,0 @@
|
|||||||
{
|
|
||||||
"env": {
|
|
||||||
"browser": true,
|
|
||||||
"es6": true,
|
|
||||||
"node": true
|
|
||||||
},
|
|
||||||
"extends": [
|
|
||||||
"eslint:recommended",
|
|
||||||
"plugin:react/recommended"
|
|
||||||
],
|
|
||||||
"settings": {
|
|
||||||
"react": {
|
|
||||||
"version": "^16.5.2"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"parser": "babel-eslint",
|
|
||||||
"parserOptions": {
|
|
||||||
"ecmaVersion": 10,
|
|
||||||
"requireConfigFile": false,
|
|
||||||
"sourceType": "module"
|
|
||||||
},
|
|
||||||
"root": true,
|
|
||||||
"rules": {
|
|
||||||
"array-bracket-spacing": ["error", "never"],
|
|
||||||
"arrow-parens": [
|
|
||||||
"error",
|
|
||||||
"as-needed",
|
|
||||||
{
|
|
||||||
"requireForBlockBody": true
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"arrow-spacing": "error",
|
|
||||||
"block-spacing": ["error", "always"],
|
|
||||||
"brace-style": ["error", "1tbs"],
|
|
||||||
"camelcase": [
|
|
||||||
"error",
|
|
||||||
{
|
|
||||||
"properties": "never"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"comma-dangle": ["error", "never"],
|
|
||||||
"eol-last": ["error", "always"],
|
|
||||||
"func-name-matching": "error",
|
|
||||||
"indent": [
|
|
||||||
"off",
|
|
||||||
2,
|
|
||||||
{
|
|
||||||
"ArrayExpression": "off",
|
|
||||||
"SwitchCase": 1,
|
|
||||||
"CallExpression": {
|
|
||||||
"arguments": "off"
|
|
||||||
},
|
|
||||||
"FunctionDeclaration": {
|
|
||||||
"parameters": "off"
|
|
||||||
},
|
|
||||||
"FunctionExpression": {
|
|
||||||
"parameters": "off"
|
|
||||||
},
|
|
||||||
"MemberExpression": "off",
|
|
||||||
"ObjectExpression": "off",
|
|
||||||
"ImportDeclaration": "off"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"handle-callback-err": "off",
|
|
||||||
"linebreak-style": ["error", "unix"],
|
|
||||||
"max-statements-per-line": [
|
|
||||||
"error",
|
|
||||||
{
|
|
||||||
"max": 1
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"new-cap": [
|
|
||||||
"error",
|
|
||||||
{
|
|
||||||
"newIsCap": true,
|
|
||||||
"capIsNew": false
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"new-parens": "error",
|
|
||||||
"no-buffer-constructor": "error",
|
|
||||||
"no-console": "off",
|
|
||||||
"no-extra-semi": "off",
|
|
||||||
"no-fallthrough": "off",
|
|
||||||
"no-func-assign": "off",
|
|
||||||
"no-implicit-coercion": "error",
|
|
||||||
"no-multi-assign": "error",
|
|
||||||
"no-multiple-empty-lines": [
|
|
||||||
"error",
|
|
||||||
{
|
|
||||||
"max": 1
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"no-nested-ternary": "error",
|
|
||||||
"no-param-reassign": "off",
|
|
||||||
"no-return-assign": "error",
|
|
||||||
"no-return-await": "off",
|
|
||||||
"no-shadow-restricted-names": "error",
|
|
||||||
"no-tabs": "error",
|
|
||||||
"no-trailing-spaces": "error",
|
|
||||||
"no-unused-vars": [
|
|
||||||
"error",
|
|
||||||
{
|
|
||||||
"vars": "all",
|
|
||||||
"args": "none",
|
|
||||||
"ignoreRestSiblings": false
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"no-use-before-define": [
|
|
||||||
"error",
|
|
||||||
{
|
|
||||||
"functions": false,
|
|
||||||
"classes": false
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"no-useless-escape": "off",
|
|
||||||
"no-var": "error",
|
|
||||||
"nonblock-statement-body-position": ["error", "below"],
|
|
||||||
"object-curly-spacing": ["error", "always"],
|
|
||||||
"padded-blocks": ["error", "never"],
|
|
||||||
"prefer-arrow-callback": "error",
|
|
||||||
"prefer-const": [
|
|
||||||
"error",
|
|
||||||
{
|
|
||||||
"destructuring": "all",
|
|
||||||
"ignoreReadBeforeAssign": true
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"prefer-template": "off",
|
|
||||||
"quotes": ["error", "single"],
|
|
||||||
"semi": ["error", "always"],
|
|
||||||
"spaced-comment": [
|
|
||||||
"error",
|
|
||||||
"always",
|
|
||||||
{
|
|
||||||
"exceptions": ["!"]
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"space-before-blocks": "error",
|
|
||||||
"unicode-bom": ["error", "never"],
|
|
||||||
"valid-jsdoc": "error",
|
|
||||||
"wrap-iife": ["error", "inside"],
|
|
||||||
"react/jsx-closing-bracket-location": 1,
|
|
||||||
"react/jsx-tag-spacing": 1,
|
|
||||||
"react/jsx-max-props-per-line": ["error", { "maximum": 2, "when": "multiline" }],
|
|
||||||
"react/prop-types": 0
|
|
||||||
}
|
|
||||||
}
|
|
@ -52,7 +52,7 @@ if(urbitrc.URL) {
|
|||||||
...devServer,
|
...devServer,
|
||||||
index: '',
|
index: '',
|
||||||
proxy: {
|
proxy: {
|
||||||
'/~landscape/js/index.js': {
|
'/~landscape/js/bundle/index.*.js': {
|
||||||
target: 'http://localhost:9000',
|
target: 'http://localhost:9000',
|
||||||
pathRewrite: (req, path) => '/index.js'
|
pathRewrite: (req, path) => '/index.js'
|
||||||
},
|
},
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
const path = require('path');
|
const path = require('path');
|
||||||
// const HtmlWebpackPlugin = require('html-webpack-plugin');
|
// const HtmlWebpackPlugin = require('html-webpack-plugin');
|
||||||
// const { CleanWebpackPlugin } = require('clean-webpack-plugin');
|
const { CleanWebpackPlugin } = require('clean-webpack-plugin');
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
mode: 'production',
|
mode: 'production',
|
||||||
@ -49,17 +49,16 @@ module.exports = {
|
|||||||
// historyApiFallback: true
|
// historyApiFallback: true
|
||||||
// },
|
// },
|
||||||
plugins: [
|
plugins: [
|
||||||
// new CleanWebpackPlugin(),
|
new CleanWebpackPlugin(),
|
||||||
// new HtmlWebpackPlugin({
|
// new HtmlWebpackPlugin({
|
||||||
// title: 'Hot Module Replacement',
|
// title: 'Hot Module Replacement',
|
||||||
// template: './public/index.html',
|
// template: './public/index.html',
|
||||||
// }),
|
// }),
|
||||||
],
|
],
|
||||||
output: {
|
output: {
|
||||||
filename: 'index.js',
|
filename: 'index.[contenthash].js',
|
||||||
chunkFilename: 'index.js',
|
path: path.resolve(__dirname, '../../arvo/app/landscape/js/bundle'),
|
||||||
path: path.resolve(__dirname, '../../arvo/app/landscape/js'),
|
publicPath: '/',
|
||||||
publicPath: '/'
|
|
||||||
},
|
},
|
||||||
optimization: {
|
optimization: {
|
||||||
minimize: true,
|
minimize: true,
|
||||||
|
226
pkg/interface/package-lock.json
generated
226
pkg/interface/package-lock.json
generated
@ -1660,6 +1660,12 @@
|
|||||||
"integrity": "sha512-rr+OQyAjxze7GgWrSaJwydHStIhHq2lvY3BOC2Mj7KnzI7XK0Uw1TOOdI9lDoajEbSWLiYgoo4f1R51erQfhPQ==",
|
"integrity": "sha512-rr+OQyAjxze7GgWrSaJwydHStIhHq2lvY3BOC2Mj7KnzI7XK0Uw1TOOdI9lDoajEbSWLiYgoo4f1R51erQfhPQ==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
|
"@types/eslint-visitor-keys": {
|
||||||
|
"version": "1.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/@types/eslint-visitor-keys/-/eslint-visitor-keys-1.0.0.tgz",
|
||||||
|
"integrity": "sha512-OCutwjDZ4aFS6PB1UZ988C4YgwlBHJd6wCeQqaLdmadZ/7e+w79+hbMUFC1QXDNCmdyoRfAFdm0RypzwR+Qpag==",
|
||||||
|
"dev": true
|
||||||
|
},
|
||||||
"@types/events": {
|
"@types/events": {
|
||||||
"version": "3.0.0",
|
"version": "3.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/@types/events/-/events-3.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/@types/events/-/events-3.0.0.tgz",
|
||||||
@ -1689,6 +1695,12 @@
|
|||||||
"integrity": "sha512-iYCgjm1dGPRuo12+BStjd1HiVQqhlRhWDOQigNxn023HcjnhsiFz9pc6CzJj4HwDCSQca9bxTL4PxJDbkdm3PA==",
|
"integrity": "sha512-iYCgjm1dGPRuo12+BStjd1HiVQqhlRhWDOQigNxn023HcjnhsiFz9pc6CzJj4HwDCSQca9bxTL4PxJDbkdm3PA==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
|
"@types/json-schema": {
|
||||||
|
"version": "7.0.5",
|
||||||
|
"resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.5.tgz",
|
||||||
|
"integrity": "sha512-7+2BITlgjgDhH0vvwZU/HZJVyk+2XUlvxXe8dFMedNX/aMkaOq++rMAFXc0tM7ij15QaWlbdQASBR9dihi+bDQ==",
|
||||||
|
"dev": true
|
||||||
|
},
|
||||||
"@types/lodash": {
|
"@types/lodash": {
|
||||||
"version": "4.14.155",
|
"version": "4.14.155",
|
||||||
"resolved": "https://registry.npmjs.org/@types/lodash/-/lodash-4.14.155.tgz",
|
"resolved": "https://registry.npmjs.org/@types/lodash/-/lodash-4.14.155.tgz",
|
||||||
@ -1795,6 +1807,93 @@
|
|||||||
"source-map": "^0.6.1"
|
"source-map": "^0.6.1"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"@typescript-eslint/eslint-plugin": {
|
||||||
|
"version": "3.8.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-3.8.0.tgz",
|
||||||
|
"integrity": "sha512-lFb4VCDleFSR+eo4Ew+HvrJ37ZH1Y9ZyE+qyP7EiwBpcCVxwmUc5PAqhShCQ8N8U5vqYydm74nss+a0wrrCErw==",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"@typescript-eslint/experimental-utils": "3.8.0",
|
||||||
|
"debug": "^4.1.1",
|
||||||
|
"functional-red-black-tree": "^1.0.1",
|
||||||
|
"regexpp": "^3.0.0",
|
||||||
|
"semver": "^7.3.2",
|
||||||
|
"tsutils": "^3.17.1"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"semver": {
|
||||||
|
"version": "7.3.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz",
|
||||||
|
"integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==",
|
||||||
|
"dev": true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"@typescript-eslint/experimental-utils": {
|
||||||
|
"version": "3.8.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/@typescript-eslint/experimental-utils/-/experimental-utils-3.8.0.tgz",
|
||||||
|
"integrity": "sha512-o8T1blo1lAJE0QDsW7nSyvZHbiDzQDjINJKyB44Z3sSL39qBy5L10ScI/XwDtaiunoyKGLiY9bzRk4YjsUZl8w==",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"@types/json-schema": "^7.0.3",
|
||||||
|
"@typescript-eslint/types": "3.8.0",
|
||||||
|
"@typescript-eslint/typescript-estree": "3.8.0",
|
||||||
|
"eslint-scope": "^5.0.0",
|
||||||
|
"eslint-utils": "^2.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"@typescript-eslint/parser": {
|
||||||
|
"version": "3.8.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-3.8.0.tgz",
|
||||||
|
"integrity": "sha512-u5vjOBaCsnMVQOvkKCXAmmOhyyMmFFf5dbkM3TIbg3MZ2pyv5peE4gj81UAbTHwTOXEwf7eCQTUMKrDl/+qGnA==",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"@types/eslint-visitor-keys": "^1.0.0",
|
||||||
|
"@typescript-eslint/experimental-utils": "3.8.0",
|
||||||
|
"@typescript-eslint/types": "3.8.0",
|
||||||
|
"@typescript-eslint/typescript-estree": "3.8.0",
|
||||||
|
"eslint-visitor-keys": "^1.1.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"@typescript-eslint/types": {
|
||||||
|
"version": "3.8.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-3.8.0.tgz",
|
||||||
|
"integrity": "sha512-8kROmEQkv6ss9kdQ44vCN1dTrgu4Qxrd2kXr10kz2NP5T8/7JnEfYNxCpPkArbLIhhkGLZV3aVMplH1RXQRF7Q==",
|
||||||
|
"dev": true
|
||||||
|
},
|
||||||
|
"@typescript-eslint/typescript-estree": {
|
||||||
|
"version": "3.8.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-3.8.0.tgz",
|
||||||
|
"integrity": "sha512-MTv9nPDhlKfclwnplRNDL44mP2SY96YmPGxmMbMy6x12I+pERcxpIUht7DXZaj4mOKKtet53wYYXU0ABaiXrLw==",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"@typescript-eslint/types": "3.8.0",
|
||||||
|
"@typescript-eslint/visitor-keys": "3.8.0",
|
||||||
|
"debug": "^4.1.1",
|
||||||
|
"glob": "^7.1.6",
|
||||||
|
"is-glob": "^4.0.1",
|
||||||
|
"lodash": "^4.17.15",
|
||||||
|
"semver": "^7.3.2",
|
||||||
|
"tsutils": "^3.17.1"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"semver": {
|
||||||
|
"version": "7.3.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz",
|
||||||
|
"integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==",
|
||||||
|
"dev": true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"@typescript-eslint/visitor-keys": {
|
||||||
|
"version": "3.8.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-3.8.0.tgz",
|
||||||
|
"integrity": "sha512-gfqQWyVPpT9NpLREXNR820AYwgz+Kr1GuF3nf1wxpHD6hdxI62tq03ToomFnDxY0m3pUB39IF7sil7D5TQexLA==",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"eslint-visitor-keys": "^1.1.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"@webassemblyjs/ast": {
|
"@webassemblyjs/ast": {
|
||||||
"version": "1.9.0",
|
"version": "1.9.0",
|
||||||
"resolved": "https://registry.npmjs.org/@webassemblyjs/ast/-/ast-1.9.0.tgz",
|
"resolved": "https://registry.npmjs.org/@webassemblyjs/ast/-/ast-1.9.0.tgz",
|
||||||
@ -1993,9 +2092,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"acorn": {
|
"acorn": {
|
||||||
"version": "7.1.1",
|
"version": "7.4.0",
|
||||||
"resolved": "https://registry.npmjs.org/acorn/-/acorn-7.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/acorn/-/acorn-7.4.0.tgz",
|
||||||
"integrity": "sha512-add7dgA5ppRPxCFJoAGfMDi7PIBXq1RtGo7BhbLaxwrXPOmw8gq48Y9ozT01hUKy9byMjlR20EJhu5zlkErEkg==",
|
"integrity": "sha512-+G7P8jJmCHr+S+cLfQxygbWhXy+8YTVGzAkpEbcLo2mLoL7tij/VG41QSHACSf5QgYRhMZYHuNc6drJaO0Da+w==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
"acorn-jsx": {
|
"acorn-jsx": {
|
||||||
@ -2931,9 +3030,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"cli-width": {
|
"cli-width": {
|
||||||
"version": "2.2.1",
|
"version": "3.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/cli-width/-/cli-width-2.2.1.tgz",
|
"resolved": "https://registry.npmjs.org/cli-width/-/cli-width-3.0.0.tgz",
|
||||||
"integrity": "sha512-GRMWDxpOB6Dgk2E5Uo+3eEBvtOOlimMmpbFiKuLFnQzYDavtLFY3K5ona41jgN/WdRZtG7utuVSVTL4HbZHGkw==",
|
"integrity": "sha512-FxqpkPPwu1HjuN93Omfm4h8uIanXofW0RxVEW3k5RKx+mJJYSthzNhp32Kzxxy3YAEZ/Dc/EWN1vZRY0+kOhbw==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
"cliui": {
|
"cliui": {
|
||||||
@ -3960,6 +4059,15 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"eslint-utils": {
|
||||||
|
"version": "1.4.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-1.4.3.tgz",
|
||||||
|
"integrity": "sha512-fbBN5W2xdY45KulGXmLHZ3c3FHfVYmKg0IrAKGOkT/464PQsx2UeIzfz1RmEci+KLm1bBaAzZAh8+/E+XAeZ8Q==",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"eslint-visitor-keys": "^1.1.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"globals": {
|
"globals": {
|
||||||
"version": "12.4.0",
|
"version": "12.4.0",
|
||||||
"resolved": "https://registry.npmjs.org/globals/-/globals-12.4.0.tgz",
|
"resolved": "https://registry.npmjs.org/globals/-/globals-12.4.0.tgz",
|
||||||
@ -3975,6 +4083,12 @@
|
|||||||
"integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=",
|
"integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
|
"regexpp": {
|
||||||
|
"version": "2.0.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/regexpp/-/regexpp-2.0.1.tgz",
|
||||||
|
"integrity": "sha512-lv0M6+TkDVniA3aD1Eg0DVpfU/booSu7Eev3TDO/mZKHBfVjgCGTV4t4buppESEYDtkArYFOxTJWv6S5C+iaNw==",
|
||||||
|
"dev": true
|
||||||
|
},
|
||||||
"shebang-command": {
|
"shebang-command": {
|
||||||
"version": "1.2.0",
|
"version": "1.2.0",
|
||||||
"resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz",
|
"resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz",
|
||||||
@ -4033,9 +4147,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"eslint-scope": {
|
"eslint-scope": {
|
||||||
"version": "5.0.0",
|
"version": "5.1.0",
|
||||||
"resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.0.tgz",
|
||||||
"integrity": "sha512-oYrhJW7S0bxAFDvWqzvMPRm6pcgcnWc4QnofCAqRTRfQC0JcwenzGglTtsLyIuuWFfkqDG9vz67cnttSd53djw==",
|
"integrity": "sha512-iiGRvtxWqgtx5m8EyQUJihBloE4EnYeGE/bz1wSPwJE6tZuJUtHlhqDM4Xj2ukE8Dyy1+HCZ4hE0fzIVMzb58w==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
"esrecurse": "^4.1.0",
|
"esrecurse": "^4.1.0",
|
||||||
@ -4043,9 +4157,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"eslint-utils": {
|
"eslint-utils": {
|
||||||
"version": "1.4.3",
|
"version": "2.1.0",
|
||||||
"resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-1.4.3.tgz",
|
"resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-2.1.0.tgz",
|
||||||
"integrity": "sha512-fbBN5W2xdY45KulGXmLHZ3c3FHfVYmKg0IrAKGOkT/464PQsx2UeIzfz1RmEci+KLm1bBaAzZAh8+/E+XAeZ8Q==",
|
"integrity": "sha512-w94dQYoauyvlDc43XnGB8lU3Zt713vNChgt4EWwhXAP2XkBvndfxF0AgIqKOOasjPIPzj9JqgwkwbCYD0/V3Zg==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
"eslint-visitor-keys": "^1.1.0"
|
"eslint-visitor-keys": "^1.1.0"
|
||||||
@ -4084,9 +4198,9 @@
|
|||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"estraverse": {
|
"estraverse": {
|
||||||
"version": "5.1.0",
|
"version": "5.2.0",
|
||||||
"resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.1.0.tgz",
|
"resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.2.0.tgz",
|
||||||
"integrity": "sha512-FyohXK+R0vE+y1nHLoBM7ZTyqRpqAlhdZHCWIWEviFLiGB8b04H6bQs8G+XTthacvT8VuwvteiP7RJSxMs8UEw==",
|
"integrity": "sha512-BxbNGGNm0RyRYvUdHpIwv9IWzeM9XClbOxwoATuFdOE7ZE6wHL+HQ5T8hoPM+zHvmKzzsEqhgy0GrQ5X13afiQ==",
|
||||||
"dev": true
|
"dev": true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -5409,21 +5523,21 @@
|
|||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
"inquirer": {
|
"inquirer": {
|
||||||
"version": "7.1.0",
|
"version": "7.3.3",
|
||||||
"resolved": "https://registry.npmjs.org/inquirer/-/inquirer-7.1.0.tgz",
|
"resolved": "https://registry.npmjs.org/inquirer/-/inquirer-7.3.3.tgz",
|
||||||
"integrity": "sha512-5fJMWEmikSYu0nv/flMc475MhGbB7TSPd/2IpFV4I4rMklboCH2rQjYY5kKiYGHqUF9gvaambupcJFFG9dvReg==",
|
"integrity": "sha512-JG3eIAj5V9CwcGvuOmoo6LB9kbAYT8HXffUl6memuszlwDC/qvFAJw49XJ5NROSFNPxp3iQg1GqkFhaY/CR0IA==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
"ansi-escapes": "^4.2.1",
|
"ansi-escapes": "^4.2.1",
|
||||||
"chalk": "^3.0.0",
|
"chalk": "^4.1.0",
|
||||||
"cli-cursor": "^3.1.0",
|
"cli-cursor": "^3.1.0",
|
||||||
"cli-width": "^2.0.0",
|
"cli-width": "^3.0.0",
|
||||||
"external-editor": "^3.0.3",
|
"external-editor": "^3.0.3",
|
||||||
"figures": "^3.0.0",
|
"figures": "^3.0.0",
|
||||||
"lodash": "^4.17.15",
|
"lodash": "^4.17.19",
|
||||||
"mute-stream": "0.0.8",
|
"mute-stream": "0.0.8",
|
||||||
"run-async": "^2.4.0",
|
"run-async": "^2.4.0",
|
||||||
"rxjs": "^6.5.3",
|
"rxjs": "^6.6.0",
|
||||||
"string-width": "^4.1.0",
|
"string-width": "^4.1.0",
|
||||||
"strip-ansi": "^6.0.0",
|
"strip-ansi": "^6.0.0",
|
||||||
"through": "^2.3.6"
|
"through": "^2.3.6"
|
||||||
@ -5440,9 +5554,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"chalk": {
|
"chalk": {
|
||||||
"version": "3.0.0",
|
"version": "4.1.0",
|
||||||
"resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz",
|
||||||
"integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==",
|
"integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
"ansi-styles": "^4.1.0",
|
"ansi-styles": "^4.1.0",
|
||||||
@ -5470,6 +5584,12 @@
|
|||||||
"integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
|
"integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
|
"lodash": {
|
||||||
|
"version": "4.17.19",
|
||||||
|
"resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.19.tgz",
|
||||||
|
"integrity": "sha512-JNvd8XER9GQX0v2qJgsaN/mzFCNA5BRe/j8JN9d+tWyGLSodKQHKFicdwNYzWwI3wjRnaKPsGj1XkBjx/F96DQ==",
|
||||||
|
"dev": true
|
||||||
|
},
|
||||||
"strip-ansi": {
|
"strip-ansi": {
|
||||||
"version": "6.0.0",
|
"version": "6.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz",
|
||||||
@ -5792,9 +5912,9 @@
|
|||||||
"integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ=="
|
"integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ=="
|
||||||
},
|
},
|
||||||
"js-yaml": {
|
"js-yaml": {
|
||||||
"version": "3.13.1",
|
"version": "3.14.0",
|
||||||
"resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.13.1.tgz",
|
"resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.0.tgz",
|
||||||
"integrity": "sha512-YfbcO7jXDdyj0DGxYVSlSeQNHbD7XPWvrVWeVUujrQEoZzWJIRrCPoyk6kL6IAjAG2IolMK4T0hNUe0HOUs5Jw==",
|
"integrity": "sha512-/4IbIeHcD9VMHFqDR/gQ7EdZdLimOvW2DdcxFjdyyZ9NsbS+ccrXqVWDtab/lRl5AlUqmpBx8EhPaWR+OtY17A==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
"argparse": "^1.0.7",
|
"argparse": "^1.0.7",
|
||||||
@ -6380,6 +6500,11 @@
|
|||||||
"resolved": "https://registry.npmjs.org/mousetrap/-/mousetrap-1.6.5.tgz",
|
"resolved": "https://registry.npmjs.org/mousetrap/-/mousetrap-1.6.5.tgz",
|
||||||
"integrity": "sha512-QNo4kEepaIBwiT8CDhP98umTetp+JNfQYBWvC1pc6/OAibuXtRcxZ58Qz8skvEHYvURne/7R8T5VoOI7rDsEUA=="
|
"integrity": "sha512-QNo4kEepaIBwiT8CDhP98umTetp+JNfQYBWvC1pc6/OAibuXtRcxZ58Qz8skvEHYvURne/7R8T5VoOI7rDsEUA=="
|
||||||
},
|
},
|
||||||
|
"mousetrap-global-bind": {
|
||||||
|
"version": "1.1.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/mousetrap-global-bind/-/mousetrap-global-bind-1.1.0.tgz",
|
||||||
|
"integrity": "sha1-zX3pIivQZG+i4BDVTISnTCaojt0="
|
||||||
|
},
|
||||||
"move-concurrently": {
|
"move-concurrently": {
|
||||||
"version": "1.0.1",
|
"version": "1.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/move-concurrently/-/move-concurrently-1.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/move-concurrently/-/move-concurrently-1.0.1.tgz",
|
||||||
@ -6774,9 +6899,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"onetime": {
|
"onetime": {
|
||||||
"version": "5.1.0",
|
"version": "5.1.1",
|
||||||
"resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.0.tgz",
|
"resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.1.tgz",
|
||||||
"integrity": "sha512-5NcSkPHhwTVFIQN+TUqXoS5+dlElHXdpAWu9I0HP20YOtIi+aZ0Ct82jdlILDxjLEAWwvm+qj1m6aEtsDVmm6Q==",
|
"integrity": "sha512-ZpZpjcJeugQfWsfyQlshVoowIIQ1qBGSVll4rfDq6JJVO//fesjoX808hXWfBjY+ROZgpKDI5TRSRBSoJiZ8eg==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
"mimic-fn": "^2.1.0"
|
"mimic-fn": "^2.1.0"
|
||||||
@ -7661,9 +7786,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"regexpp": {
|
"regexpp": {
|
||||||
"version": "2.0.1",
|
"version": "3.1.0",
|
||||||
"resolved": "https://registry.npmjs.org/regexpp/-/regexpp-2.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/regexpp/-/regexpp-3.1.0.tgz",
|
||||||
"integrity": "sha512-lv0M6+TkDVniA3aD1Eg0DVpfU/booSu7Eev3TDO/mZKHBfVjgCGTV4t4buppESEYDtkArYFOxTJWv6S5C+iaNw==",
|
"integrity": "sha512-ZOIzd8yVsQQA7j8GCSlPGXwg5PfmA1mrq0JP4nGhh54LaKN3xdai/vHUDu74pKwV8OxseMS65u2NImosQcSD0Q==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
"regexpu-core": {
|
"regexpu-core": {
|
||||||
@ -7982,9 +8107,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"rxjs": {
|
"rxjs": {
|
||||||
"version": "6.5.5",
|
"version": "6.6.2",
|
||||||
"resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.5.5.tgz",
|
"resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.6.2.tgz",
|
||||||
"integrity": "sha512-WfQI+1gohdf0Dai/Bbmk5L5ItH5tYqm3ki2c5GdWhKjalzjg93N3avFjVStyZZz+A2Em+ZxKH5bNghw9UeylGQ==",
|
"integrity": "sha512-BHdBMVoWC2sL26w//BCu3YzKT4s2jip/WhwsGEDmeKYBhKDZeYezVUnHatYB7L85v5xs0BAQmg6BEYJEKxBabg==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
"tslib": "^1.9.0"
|
"tslib": "^1.9.0"
|
||||||
@ -8826,9 +8951,9 @@
|
|||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
"strip-json-comments": {
|
"strip-json-comments": {
|
||||||
"version": "3.1.0",
|
"version": "3.1.1",
|
||||||
"resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.0.tgz",
|
"resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz",
|
||||||
"integrity": "sha512-e6/d0eBu7gHtdCqFt0xJr642LdToM5/cN4Qb9DbHjVx1CP5RyeM+zH7pbecEmDv/lBqb0QH+6Uqq75rxFPkM0w==",
|
"integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
"style-loader": {
|
"style-loader": {
|
||||||
@ -9187,6 +9312,15 @@
|
|||||||
"resolved": "https://registry.npmjs.org/tslib/-/tslib-1.11.1.tgz",
|
"resolved": "https://registry.npmjs.org/tslib/-/tslib-1.11.1.tgz",
|
||||||
"integrity": "sha512-aZW88SY8kQbU7gpV19lN24LtXh/yD4ZZg6qieAJDDg+YBsJcSmLGK9QpnUjAKVG/xefmvJGd1WUmfpT/g6AJGA=="
|
"integrity": "sha512-aZW88SY8kQbU7gpV19lN24LtXh/yD4ZZg6qieAJDDg+YBsJcSmLGK9QpnUjAKVG/xefmvJGd1WUmfpT/g6AJGA=="
|
||||||
},
|
},
|
||||||
|
"tsutils": {
|
||||||
|
"version": "3.17.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/tsutils/-/tsutils-3.17.1.tgz",
|
||||||
|
"integrity": "sha512-kzeQ5B8H3w60nFY2g8cJIuH7JDpsALXySGtwGJ0p2LSjLgay3NdIpqq5SoOBe46bKDW2iq25irHCr8wjomUS2g==",
|
||||||
|
"dev": true,
|
||||||
|
"requires": {
|
||||||
|
"tslib": "^1.8.1"
|
||||||
|
}
|
||||||
|
},
|
||||||
"tty-browserify": {
|
"tty-browserify": {
|
||||||
"version": "0.0.0",
|
"version": "0.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/tty-browserify/-/tty-browserify-0.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/tty-browserify/-/tty-browserify-0.0.0.tgz",
|
||||||
@ -9224,6 +9358,12 @@
|
|||||||
"integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=",
|
"integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
|
"typescript": {
|
||||||
|
"version": "3.9.7",
|
||||||
|
"resolved": "https://registry.npmjs.org/typescript/-/typescript-3.9.7.tgz",
|
||||||
|
"integrity": "sha512-BLbiRkiBzAwsjut4x/dsibSTB6yWpwT5qWmC2OfuCg3GgVQCSgMs4vEctYPhsaGtd0AeuuHMkjZ2h2WG8MSzRw==",
|
||||||
|
"dev": true
|
||||||
|
},
|
||||||
"unherit": {
|
"unherit": {
|
||||||
"version": "1.1.3",
|
"version": "1.1.3",
|
||||||
"resolved": "https://registry.npmjs.org/unherit/-/unherit-1.1.3.tgz",
|
"resolved": "https://registry.npmjs.org/unherit/-/unherit-1.1.3.tgz",
|
||||||
@ -9497,9 +9637,9 @@
|
|||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
"v8-compile-cache": {
|
"v8-compile-cache": {
|
||||||
"version": "2.1.0",
|
"version": "2.1.1",
|
||||||
"resolved": "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.1.0.tgz",
|
"resolved": "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.1.1.tgz",
|
||||||
"integrity": "sha512-usZBT3PW+LOjM25wbqIlZwPeJV+3OSz3M1k1Ws8snlW39dZyYL9lOGC5FgPVHfk0jKmjiDV8Z0mIbVQPiwFs7g==",
|
"integrity": "sha512-8OQ9CL+VWyt3JStj7HX7/ciTL2V3Rl1Wf5OL+SNTm0yK1KvtReVulksyeRnCANHHuUxHlQig+JJDlUhBt1NQDQ==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
"value-equal": {
|
"value-equal": {
|
||||||
|
@ -18,6 +18,7 @@
|
|||||||
"markdown-to-jsx": "^6.11.4",
|
"markdown-to-jsx": "^6.11.4",
|
||||||
"moment": "^2.20.1",
|
"moment": "^2.20.1",
|
||||||
"mousetrap": "^1.6.5",
|
"mousetrap": "^1.6.5",
|
||||||
|
"mousetrap-global-bind": "^1.1.0",
|
||||||
"prop-types": "^15.7.2",
|
"prop-types": "^15.7.2",
|
||||||
"react": "^16.5.2",
|
"react": "^16.5.2",
|
||||||
"react-codemirror2": "^6.0.1",
|
"react-codemirror2": "^6.0.1",
|
||||||
@ -45,6 +46,8 @@
|
|||||||
"@types/lodash": "^4.14.155",
|
"@types/lodash": "^4.14.155",
|
||||||
"@types/react": "^16.9.38",
|
"@types/react": "^16.9.38",
|
||||||
"@types/react-router-dom": "^5.1.5",
|
"@types/react-router-dom": "^5.1.5",
|
||||||
|
"@typescript-eslint/eslint-plugin": "^3.8.0",
|
||||||
|
"@typescript-eslint/parser": "^3.8.0",
|
||||||
"babel-eslint": "^10.1.0",
|
"babel-eslint": "^10.1.0",
|
||||||
"babel-loader": "^8.1.0",
|
"babel-loader": "^8.1.0",
|
||||||
"clean-webpack-plugin": "^3.0.0",
|
"clean-webpack-plugin": "^3.0.0",
|
||||||
@ -56,12 +59,13 @@
|
|||||||
"react-hot-loader": "^4.12.21",
|
"react-hot-loader": "^4.12.21",
|
||||||
"sass": "^1.26.5",
|
"sass": "^1.26.5",
|
||||||
"sass-loader": "^8.0.2",
|
"sass-loader": "^8.0.2",
|
||||||
|
"typescript": "^3.9.7",
|
||||||
"webpack": "^4.43.0",
|
"webpack": "^4.43.0",
|
||||||
"webpack-cli": "^3.3.11",
|
"webpack-cli": "^3.3.11",
|
||||||
"webpack-dev-server": "^3.10.3"
|
"webpack-dev-server": "^3.10.3"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"lint": "eslint ./**/*.js",
|
"lint": "eslint ./src/**/*.{js,ts,tsx}",
|
||||||
"lint-file": "eslint",
|
"lint-file": "eslint",
|
||||||
"tsc": "tsc",
|
"tsc": "tsc",
|
||||||
"tsc:watch": "tsc --watch",
|
"tsc:watch": "tsc --watch",
|
||||||
|
@ -3,6 +3,10 @@ import 'react-hot-loader';
|
|||||||
import * as React from 'react';
|
import * as React from 'react';
|
||||||
import { BrowserRouter as Router, Route, withRouter, Switch } from 'react-router-dom';
|
import { BrowserRouter as Router, Route, withRouter, Switch } from 'react-router-dom';
|
||||||
import styled, { ThemeProvider, createGlobalStyle } from 'styled-components';
|
import styled, { ThemeProvider, createGlobalStyle } from 'styled-components';
|
||||||
|
import { sigil as sigiljs, stringRenderer } from 'urbit-sigil-js';
|
||||||
|
|
||||||
|
import Mousetrap from 'mousetrap';
|
||||||
|
import 'mousetrap-global-bind';
|
||||||
|
|
||||||
import './css/indigo-static.css';
|
import './css/indigo-static.css';
|
||||||
import './css/fonts.css';
|
import './css/fonts.css';
|
||||||
@ -17,11 +21,14 @@ import LinksApp from './apps/links/app';
|
|||||||
import PublishApp from './apps/publish/app';
|
import PublishApp from './apps/publish/app';
|
||||||
|
|
||||||
import StatusBar from './components/StatusBar';
|
import StatusBar from './components/StatusBar';
|
||||||
|
import Omnibox from './components/Omnibox';
|
||||||
import ErrorComponent from './components/Error';
|
import ErrorComponent from './components/Error';
|
||||||
|
|
||||||
import GlobalStore from './store/store';
|
import GlobalStore from './store/store';
|
||||||
import GlobalSubscription from './subscription/global';
|
import GlobalSubscription from './subscription/global';
|
||||||
import GlobalApi from './api/global';
|
import GlobalApi from './api/global';
|
||||||
|
import { uxToHex } from './lib/util';
|
||||||
|
import { Sigil } from './lib/sigil';
|
||||||
|
|
||||||
// const Style = createGlobalStyle`
|
// const Style = createGlobalStyle`
|
||||||
// ${cssReset}
|
// ${cssReset}
|
||||||
@ -62,6 +69,7 @@ class App extends React.Component {
|
|||||||
new GlobalSubscription(this.store, this.api, this.appChannel);
|
new GlobalSubscription(this.store, this.api, this.appChannel);
|
||||||
|
|
||||||
this.updateTheme = this.updateTheme.bind(this);
|
this.updateTheme = this.updateTheme.bind(this);
|
||||||
|
this.setFavicon = this.setFavicon.bind(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
componentDidMount() {
|
componentDidMount() {
|
||||||
@ -70,21 +78,49 @@ class App extends React.Component {
|
|||||||
this.api.local.setDark(this.themeWatcher.matches);
|
this.api.local.setDark(this.themeWatcher.matches);
|
||||||
this.themeWatcher.addListener(this.updateTheme);
|
this.themeWatcher.addListener(this.updateTheme);
|
||||||
this.api.local.getBaseHash();
|
this.api.local.getBaseHash();
|
||||||
|
Mousetrap.bindGlobal(['command+/', 'ctrl+/'], (e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
this.api.local.setOmnibox();
|
||||||
|
});
|
||||||
|
this.setFavicon();
|
||||||
}
|
}
|
||||||
|
|
||||||
componentWillUnmount() {
|
componentWillUnmount() {
|
||||||
this.themeWatcher.removeListener(this.updateTheme);
|
this.themeWatcher.removeListener(this.updateTheme);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
componentDidUpdate(prevProps, prevState, snapshot) {
|
||||||
|
this.setFavicon();
|
||||||
|
}
|
||||||
|
|
||||||
updateTheme(e) {
|
updateTheme(e) {
|
||||||
this.api.local.setDark(e.matches);
|
this.api.local.setDark(e.matches);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
setFavicon() {
|
||||||
|
if (window.ship.length < 14) {
|
||||||
|
let background = '#ffffff';
|
||||||
|
if (this.state.contacts.hasOwnProperty('/~/default')) {
|
||||||
|
background = `#${uxToHex(this.state.contacts['/~/default'][window.ship].color)}`;
|
||||||
|
}
|
||||||
|
const foreground = Sigil.foregroundFromBackground(background);
|
||||||
|
const svg = sigiljs({
|
||||||
|
patp: window.ship,
|
||||||
|
renderer: stringRenderer,
|
||||||
|
size: 16,
|
||||||
|
colors: [background, foreground]
|
||||||
|
});
|
||||||
|
const dataurl = 'data:image/svg+xml;base64,' + btoa(svg);
|
||||||
|
const favicon = document.querySelector('[rel=icon]');
|
||||||
|
favicon.href = dataurl;
|
||||||
|
favicon.type = 'image/svg+xml';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
const channel = window.channel;
|
const channel = window.channel;
|
||||||
|
|
||||||
const associations = this.state.associations ? this.state.associations : { contacts: {} };
|
const associations = this.state.associations ? this.state.associations : { contacts: {} };
|
||||||
const selectedGroups = this.state.selectedGroups ? this.state.selectedGroups : [];
|
|
||||||
const { state } = this;
|
const { state } = this;
|
||||||
const theme = state.dark ? dark : light;
|
const theme = state.dark ? dark : light;
|
||||||
|
|
||||||
@ -92,81 +128,99 @@ class App extends React.Component {
|
|||||||
<ThemeProvider theme={theme}>
|
<ThemeProvider theme={theme}>
|
||||||
<Root>
|
<Root>
|
||||||
<Router>
|
<Router>
|
||||||
<StatusBarWithRouter props={this.props}
|
<StatusBarWithRouter
|
||||||
associations={associations}
|
props={this.props}
|
||||||
invites={this.state.invites}
|
associations={associations}
|
||||||
api={this.api}
|
invites={this.state.invites}
|
||||||
connection={this.state.connection}
|
api={this.api}
|
||||||
subscription={this.subscription}
|
connection={this.state.connection}
|
||||||
|
subscription={this.subscription}
|
||||||
|
/>
|
||||||
|
<Omnibox
|
||||||
|
associations={state.associations}
|
||||||
|
apps={state.launch}
|
||||||
|
api={this.api}
|
||||||
|
dark={state.dark}
|
||||||
|
show={state.omniboxShown}
|
||||||
/>
|
/>
|
||||||
<Content>
|
<Content>
|
||||||
<Switch>
|
<Switch>
|
||||||
<Route exact path="/"
|
<Route
|
||||||
render={ p => (
|
exact
|
||||||
<LaunchApp
|
path='/'
|
||||||
ship={this.ship}
|
render={p => (
|
||||||
api={this.api}
|
<LaunchApp
|
||||||
{...state}
|
ship={this.ship}
|
||||||
{...p}
|
api={this.api}
|
||||||
|
{...state}
|
||||||
|
{...p}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
/>
|
/>
|
||||||
)}
|
<Route
|
||||||
/>
|
path='/~chat'
|
||||||
<Route path="/~chat" render={ p => (
|
render={p => (
|
||||||
<ChatApp
|
<ChatApp
|
||||||
ship={this.ship}
|
ship={this.ship}
|
||||||
api={this.api}
|
api={this.api}
|
||||||
subscription={this.subscription}
|
subscription={this.subscription}
|
||||||
{...state}
|
{...state}
|
||||||
{...p}
|
{...p}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
/>
|
/>
|
||||||
)}
|
<Route
|
||||||
/>
|
path='/~dojo'
|
||||||
<Route path="/~dojo" render={ p => (
|
render={p => (
|
||||||
<DojoApp
|
<DojoApp
|
||||||
ship={this.ship}
|
ship={this.ship}
|
||||||
channel={channel}
|
channel={channel}
|
||||||
selectedGroups={selectedGroups}
|
subscription={this.subscription}
|
||||||
subscription={this.subscription}
|
{...p}
|
||||||
{...p}
|
/>
|
||||||
|
)}
|
||||||
/>
|
/>
|
||||||
)}
|
<Route
|
||||||
/>
|
path='/~groups'
|
||||||
<Route path="/~groups" render={ p => (
|
render={p => (
|
||||||
<GroupsApp
|
<GroupsApp
|
||||||
ship={this.ship}
|
ship={this.ship}
|
||||||
api={this.api}
|
api={this.api}
|
||||||
subscription={this.subscription}
|
subscription={this.subscription}
|
||||||
{...state}
|
{...state}
|
||||||
{...p}
|
{...p}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
/>
|
/>
|
||||||
)}
|
<Route
|
||||||
/>
|
path='/~link'
|
||||||
<Route path="/~link" render={ p => (
|
render={p => (
|
||||||
<LinksApp
|
<LinksApp
|
||||||
ship={this.ship}
|
ship={this.ship}
|
||||||
ship={this.ship}
|
api={this.api}
|
||||||
api={this.api}
|
subscription={this.subscription}
|
||||||
subscription={this.subscription}
|
{...state}
|
||||||
{...state}
|
{...p}
|
||||||
{...p}
|
/>
|
||||||
|
)}
|
||||||
/>
|
/>
|
||||||
)}
|
<Route
|
||||||
/>
|
path='/~publish'
|
||||||
<Route path="/~publish" render={ p => (
|
render={p => (
|
||||||
<PublishApp
|
<PublishApp
|
||||||
ship={this.ship}
|
ship={this.ship}
|
||||||
api={this.api}
|
api={this.api}
|
||||||
subscription={this.subscription}
|
subscription={this.subscription}
|
||||||
{...state}
|
{...state}
|
||||||
{...p}
|
{...p}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
/>
|
/>
|
||||||
)}
|
|
||||||
/>
|
|
||||||
<Route
|
<Route
|
||||||
render={(props) => (
|
render={props => (
|
||||||
<ErrorComponent {...props} code={404} description="Not Found" />
|
<ErrorComponent {...props} code={404} description="Not Found" />
|
||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
</Switch>
|
</Switch>
|
||||||
</Content>
|
</Content>
|
||||||
</Router>
|
</Router>
|
||||||
@ -176,5 +230,6 @@ class App extends React.Component {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
export default process.env.NODE_ENV === 'production' ? App : hot(App);
|
export default process.env.NODE_ENV === 'production' ? App : hot(App);
|
||||||
|
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
import BaseApi from "./base";
|
import BaseApi from "./base";
|
||||||
import { StoreState } from "../store/type";
|
import { StoreState } from "../store/type";
|
||||||
import { SelectedGroup } from "../types/local-update";
|
|
||||||
|
|
||||||
export default class LocalApi extends BaseApi<StoreState> {
|
export default class LocalApi extends BaseApi<StoreState> {
|
||||||
getBaseHash() {
|
getBaseHash() {
|
||||||
@ -9,16 +8,6 @@ export default class LocalApi extends BaseApi<StoreState> {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
setSelected(selected: SelectedGroup[]) {
|
|
||||||
this.store.handleEvent({
|
|
||||||
data: {
|
|
||||||
local: {
|
|
||||||
selected
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
sidebarToggle() {
|
sidebarToggle() {
|
||||||
this.store.handleEvent({
|
this.store.handleEvent({
|
||||||
data: {
|
data: {
|
||||||
@ -39,4 +28,14 @@ export default class LocalApi extends BaseApi<StoreState> {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
setOmnibox() {
|
||||||
|
this.store.handleEvent({
|
||||||
|
data: {
|
||||||
|
local: {
|
||||||
|
omniboxShown: true
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -53,7 +53,6 @@ export default class ChatApp extends React.Component<ChatAppProps, {}> {
|
|||||||
const unreads = {};
|
const unreads = {};
|
||||||
let totalUnreads = 0;
|
let totalUnreads = 0;
|
||||||
|
|
||||||
const selectedGroups = props.selectedGroups ? props.selectedGroups : [];
|
|
||||||
const associations = props.associations
|
const associations = props.associations
|
||||||
? props.associations
|
? props.associations
|
||||||
: { chat: {}, contacts: {} };
|
: { chat: {}, contacts: {} };
|
||||||
@ -74,14 +73,7 @@ export default class ChatApp extends React.Component<ChatAppProps, {}> {
|
|||||||
unreads[stat] = Boolean(unread);
|
unreads[stat] = Boolean(unread);
|
||||||
if (
|
if (
|
||||||
unread &&
|
unread &&
|
||||||
stat in associations.chat &&
|
stat in associations.chat
|
||||||
(selectedGroups.length === 0 ||
|
|
||||||
selectedGroups
|
|
||||||
.map((e) => {
|
|
||||||
return e[0];
|
|
||||||
})
|
|
||||||
.includes(associations.chat?.[stat]?.['group-path']) ||
|
|
||||||
props.groups[associations.chat?.[stat]?.['group-path']]?.hidden)
|
|
||||||
) {
|
) {
|
||||||
totalUnreads += unread;
|
totalUnreads += unread;
|
||||||
}
|
}
|
||||||
@ -111,7 +103,6 @@ export default class ChatApp extends React.Component<ChatAppProps, {}> {
|
|||||||
inbox={inbox}
|
inbox={inbox}
|
||||||
messagePreviews={messagePreviews}
|
messagePreviews={messagePreviews}
|
||||||
associations={associations}
|
associations={associations}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
contacts={contacts}
|
contacts={contacts}
|
||||||
invites={invites['/chat'] || {}}
|
invites={invites['/chat'] || {}}
|
||||||
unreads={unreads}
|
unreads={unreads}
|
||||||
@ -286,44 +277,6 @@ export default class ChatApp extends React.Component<ChatAppProps, {}> {
|
|||||||
);
|
);
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<Route
|
|
||||||
exact
|
|
||||||
path="/~chat/(popout)?/members/(~)?/:ship/:station+"
|
|
||||||
render={(props) => {
|
|
||||||
let station = `/${props.match.params.ship}/${props.match.params.station}`;
|
|
||||||
|
|
||||||
const popout = props.match.url.includes('/popout/');
|
|
||||||
|
|
||||||
const association =
|
|
||||||
station in associations['chat'] ? associations.chat[station] : {};
|
|
||||||
const groupPath = association['group-path'];
|
|
||||||
|
|
||||||
const group = groups[groupPath] || {};
|
|
||||||
return (
|
|
||||||
<Skeleton
|
|
||||||
associations={associations}
|
|
||||||
invites={invites}
|
|
||||||
sidebarHideOnMobile={true}
|
|
||||||
sidebarShown={sidebarShown}
|
|
||||||
popout={popout}
|
|
||||||
sidebar={renderChannelSidebar(props, station)}
|
|
||||||
>
|
|
||||||
<MemberScreen
|
|
||||||
{...props}
|
|
||||||
api={api}
|
|
||||||
group={group}
|
|
||||||
groups={groups}
|
|
||||||
associations={associations}
|
|
||||||
station={station}
|
|
||||||
association={association}
|
|
||||||
contacts={contacts}
|
|
||||||
popout={popout}
|
|
||||||
sidebarShown={sidebarShown}
|
|
||||||
/>
|
|
||||||
</Skeleton>
|
|
||||||
);
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
<Route
|
<Route
|
||||||
exact
|
exact
|
||||||
path="/~chat/(popout)?/settings/(~)?/:ship/:station+"
|
path="/~chat/(popout)?/settings/(~)?/:ship/:station+"
|
||||||
|
@ -1,16 +1,11 @@
|
|||||||
import React, { Component, Fragment } from "react";
|
import React, { Component, Fragment } from "react";
|
||||||
import _ from "lodash";
|
|
||||||
import moment from "moment";
|
import moment from "moment";
|
||||||
|
|
||||||
import { Link, RouteComponentProps } from "react-router-dom";
|
import { Link, RouteComponentProps } from "react-router-dom";
|
||||||
|
|
||||||
import { ResubscribeElement } from "./lib/resubscribe-element";
|
import { ChatWindow } from './lib/chat-window';
|
||||||
import { BacklogElement } from "./lib/backlog-element";
|
import { ChatHeader } from './lib/chat-header';
|
||||||
import { Message } from "./lib/message";
|
|
||||||
import { SidebarSwitcher } from "../../../components/SidebarSwitch";
|
|
||||||
import { ChatTabBar } from "./lib/chat-tabbar";
|
|
||||||
import { ChatInput } from "./lib/chat-input";
|
import { ChatInput } from "./lib/chat-input";
|
||||||
import { UnreadNotice } from "./lib/unread-notice";
|
|
||||||
import { deSig } from "../../../lib/util";
|
import { deSig } from "../../../lib/util";
|
||||||
import { ChatHookUpdate } from "../../../types/chat-hook-update";
|
import { ChatHookUpdate } from "../../../types/chat-hook-update";
|
||||||
import ChatApi from "../../../api/chat";
|
import ChatApi from "../../../api/chat";
|
||||||
@ -21,52 +16,6 @@ import GlobalApi from "../../../api/global";
|
|||||||
import { Association } from "../../../types/metadata-update";
|
import { Association } from "../../../types/metadata-update";
|
||||||
import {Group} from "../../../types/group-update";
|
import {Group} from "../../../types/group-update";
|
||||||
|
|
||||||
function getNumPending(props: any) {
|
|
||||||
const result = props.pendingMessages.has(props.station)
|
|
||||||
? props.pendingMessages.get(props.station).length
|
|
||||||
: 0;
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
const ACTIVITY_TIMEOUT = 60000; // a minute
|
|
||||||
const DEFAULT_BACKLOG_SIZE = 300;
|
|
||||||
const MAX_BACKLOG_SIZE = 1000;
|
|
||||||
|
|
||||||
function scrollIsAtTop(container) {
|
|
||||||
if (
|
|
||||||
(navigator.userAgent.includes("Safari") &&
|
|
||||||
navigator.userAgent.includes("Chrome")) ||
|
|
||||||
navigator.userAgent.includes("Firefox")
|
|
||||||
) {
|
|
||||||
return container.scrollTop === 0;
|
|
||||||
} else if (navigator.userAgent.includes("Safari")) {
|
|
||||||
return (
|
|
||||||
container.scrollHeight + Math.round(container.scrollTop) <=
|
|
||||||
container.clientHeight + 10
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function scrollIsAtBottom(container) {
|
|
||||||
if (
|
|
||||||
(navigator.userAgent.includes("Safari") &&
|
|
||||||
navigator.userAgent.includes("Chrome")) ||
|
|
||||||
navigator.userAgent.includes("Firefox")
|
|
||||||
) {
|
|
||||||
return (
|
|
||||||
container.scrollHeight - Math.round(container.scrollTop) <=
|
|
||||||
container.clientHeight + 10
|
|
||||||
);
|
|
||||||
} else if (navigator.userAgent.includes("Safari")) {
|
|
||||||
return container.scrollTop === 0;
|
|
||||||
} else {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
type IMessage = Envelope & { pending?: boolean };
|
|
||||||
|
|
||||||
type ChatScreenProps = RouteComponentProps<{
|
type ChatScreenProps = RouteComponentProps<{
|
||||||
ship: Patp;
|
ship: Patp;
|
||||||
@ -90,47 +39,20 @@ type ChatScreenProps = RouteComponentProps<{
|
|||||||
};
|
};
|
||||||
|
|
||||||
interface ChatScreenState {
|
interface ChatScreenState {
|
||||||
numPages: number;
|
|
||||||
scrollLocked: boolean;
|
|
||||||
read: number;
|
|
||||||
active: boolean;
|
|
||||||
messages: Map<string, string>;
|
messages: Map<string, string>;
|
||||||
lastScrollHeight: number | null;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export class ChatScreen extends Component<ChatScreenProps, ChatScreenState> {
|
export class ChatScreen extends Component<ChatScreenProps, ChatScreenState> {
|
||||||
hasAskedForMessages = false;
|
|
||||||
lastNumPending = 0;
|
lastNumPending = 0;
|
||||||
|
|
||||||
scrollContainer: HTMLElement | null = null;
|
|
||||||
|
|
||||||
unreadMarker = null;
|
|
||||||
scrolledToMarker = false;
|
|
||||||
|
|
||||||
activityTimeout: NodeJS.Timeout | null = null;
|
activityTimeout: NodeJS.Timeout | null = null;
|
||||||
|
|
||||||
scrollElement: HTMLElement | null = null;
|
|
||||||
|
|
||||||
constructor(props) {
|
constructor(props) {
|
||||||
super(props);
|
super(props);
|
||||||
|
|
||||||
this.state = {
|
this.state = {
|
||||||
numPages: 1,
|
|
||||||
scrollLocked: false,
|
|
||||||
read: props.read,
|
|
||||||
active: true,
|
|
||||||
messages: new Map(),
|
messages: new Map(),
|
||||||
// only for FF
|
|
||||||
lastScrollHeight: null,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
this.onScroll = this.onScroll.bind(this);
|
|
||||||
|
|
||||||
this.setUnreadMarker = this.setUnreadMarker.bind(this);
|
|
||||||
|
|
||||||
this.handleActivity = this.handleActivity.bind(this);
|
|
||||||
this.setInactive = this.setInactive.bind(this);
|
|
||||||
|
|
||||||
moment.updateLocale("en", {
|
moment.updateLocale("en", {
|
||||||
calendar: {
|
calendar: {
|
||||||
sameDay: "[Today]",
|
sameDay: "[Today]",
|
||||||
@ -143,450 +65,68 @@ export class ChatScreen extends Component<ChatScreenProps, ChatScreenState> {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
componentDidMount() {
|
|
||||||
document.addEventListener("mousemove", this.handleActivity, false);
|
|
||||||
document.addEventListener("mousedown", this.handleActivity, false);
|
|
||||||
document.addEventListener("keypress", this.handleActivity, false);
|
|
||||||
document.addEventListener("touchmove", this.handleActivity, false);
|
|
||||||
this.activityTimeout = setTimeout(this.setInactive, ACTIVITY_TIMEOUT);
|
|
||||||
}
|
|
||||||
|
|
||||||
componentWillUnmount() {
|
|
||||||
document.removeEventListener("mousemove", this.handleActivity, false);
|
|
||||||
document.removeEventListener("mousedown", this.handleActivity, false);
|
|
||||||
document.removeEventListener("keypress", this.handleActivity, false);
|
|
||||||
document.removeEventListener("touchmove", this.handleActivity, false);
|
|
||||||
if (this.activityTimeout) {
|
|
||||||
clearTimeout(this.activityTimeout);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
handleActivity() {
|
|
||||||
if (!this.state.active) {
|
|
||||||
this.setState({ active: true });
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.activityTimeout) {
|
|
||||||
clearTimeout(this.activityTimeout);
|
|
||||||
}
|
|
||||||
|
|
||||||
this.activityTimeout = setTimeout(this.setInactive, ACTIVITY_TIMEOUT);
|
|
||||||
}
|
|
||||||
|
|
||||||
setInactive() {
|
|
||||||
this.activityTimeout = null;
|
|
||||||
this.setState({ active: false, scrollLocked: true });
|
|
||||||
}
|
|
||||||
|
|
||||||
receivedNewChat() {
|
|
||||||
const { props } = this;
|
|
||||||
this.hasAskedForMessages = false;
|
|
||||||
|
|
||||||
this.unreadMarker = null;
|
|
||||||
this.scrolledToMarker = false;
|
|
||||||
|
|
||||||
this.setState({ read: props.read });
|
|
||||||
|
|
||||||
const unread = props.length - props.read;
|
|
||||||
const unreadUnloaded = unread - props.envelopes.length;
|
|
||||||
const excessUnread = unreadUnloaded > MAX_BACKLOG_SIZE;
|
|
||||||
|
|
||||||
if (!excessUnread && unreadUnloaded + 20 > DEFAULT_BACKLOG_SIZE) {
|
|
||||||
this.askForMessages(unreadUnloaded + 20);
|
|
||||||
} else {
|
|
||||||
this.askForMessages(DEFAULT_BACKLOG_SIZE);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (excessUnread || props.read === props.length) {
|
|
||||||
this.scrolledToMarker = true;
|
|
||||||
this.setState(
|
|
||||||
{
|
|
||||||
scrollLocked: false,
|
|
||||||
},
|
|
||||||
() => {
|
|
||||||
this.scrollToBottom();
|
|
||||||
}
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
this.setState({ scrollLocked: true, numPages: Math.ceil(unread / 100) });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
componentDidUpdate(prevProps, prevState) {
|
|
||||||
const { props, state } = this;
|
|
||||||
|
|
||||||
if (
|
|
||||||
prevProps.match.params.station !== props.match.params.station ||
|
|
||||||
prevProps.match.params.ship !== props.match.params.ship
|
|
||||||
) {
|
|
||||||
this.receivedNewChat();
|
|
||||||
} else if (
|
|
||||||
props.chatInitialized &&
|
|
||||||
!(props.station in props.inbox) &&
|
|
||||||
Boolean(props.chatSynced) &&
|
|
||||||
!(props.station in props.chatSynced)
|
|
||||||
) {
|
|
||||||
props.history.push("/~chat");
|
|
||||||
} else if (props.envelopes.length >= prevProps.envelopes.length + 10) {
|
|
||||||
this.hasAskedForMessages = false;
|
|
||||||
} else if (
|
|
||||||
props.length !== prevProps.length &&
|
|
||||||
prevProps.length === prevState.read &&
|
|
||||||
state.active
|
|
||||||
) {
|
|
||||||
this.setState({ read: props.length });
|
|
||||||
this.props.api.chat.read(this.props.station);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!prevProps.chatInitialized && props.chatInitialized) {
|
|
||||||
this.receivedNewChat();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (
|
|
||||||
props.length !== prevProps.length ||
|
|
||||||
props.envelopes.length !== prevProps.envelopes.length ||
|
|
||||||
getNumPending(props) !== this.lastNumPending ||
|
|
||||||
state.numPages !== prevState.numPages
|
|
||||||
) {
|
|
||||||
this.scrollToBottom();
|
|
||||||
if (navigator.userAgent.includes("Firefox")) {
|
|
||||||
this.recalculateScrollTop();
|
|
||||||
}
|
|
||||||
|
|
||||||
this.lastNumPending = getNumPending(props);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
askForMessages(size) {
|
|
||||||
const { props, state } = this;
|
|
||||||
|
|
||||||
if (
|
|
||||||
props.envelopes.length >= props.length ||
|
|
||||||
this.hasAskedForMessages ||
|
|
||||||
props.length <= 0
|
|
||||||
) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const start =
|
|
||||||
props.length - props.envelopes[props.envelopes.length - 1].number;
|
|
||||||
if (start > 0) {
|
|
||||||
const end = start + size < props.length ? start + size : props.length;
|
|
||||||
this.hasAskedForMessages = true;
|
|
||||||
props.api.chat.fetchMessages(start + 1, end, props.station);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
scrollToBottom() {
|
|
||||||
if (!this.state.scrollLocked && this.scrollElement) {
|
|
||||||
this.scrollElement.scrollIntoView();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Restore chat position on FF when new messages come in
|
|
||||||
recalculateScrollTop() {
|
|
||||||
const { lastScrollHeight } = this.state;
|
|
||||||
if (!this.scrollContainer || !lastScrollHeight) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const target = this.scrollContainer;
|
|
||||||
const newScrollTop = this.scrollContainer.scrollHeight - lastScrollHeight;
|
|
||||||
if (target.scrollTop !== 0 || newScrollTop === target.scrollTop) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
target.scrollTop = target.scrollHeight - lastScrollHeight;
|
|
||||||
}
|
|
||||||
|
|
||||||
onScroll(e) {
|
|
||||||
if (scrollIsAtTop(e.target)) {
|
|
||||||
// Save scroll position for FF
|
|
||||||
if (navigator.userAgent.includes("Firefox")) {
|
|
||||||
this.setState({
|
|
||||||
lastScrollHeight: e.target.scrollHeight,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
this.setState(
|
|
||||||
{
|
|
||||||
numPages: this.state.numPages + 1,
|
|
||||||
scrollLocked: true,
|
|
||||||
},
|
|
||||||
() => {
|
|
||||||
this.askForMessages(DEFAULT_BACKLOG_SIZE);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
} else if (scrollIsAtBottom(e.target)) {
|
|
||||||
this.dismissUnread();
|
|
||||||
this.setState({
|
|
||||||
numPages: 1,
|
|
||||||
scrollLocked: false,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
setUnreadMarker(ref) {
|
|
||||||
if (ref && !this.scrolledToMarker) {
|
|
||||||
this.setState({ scrollLocked: true }, () => {
|
|
||||||
ref.scrollIntoView({ block: "center" });
|
|
||||||
if (ref.offsetParent && scrollIsAtBottom(ref.offsetParent)) {
|
|
||||||
this.dismissUnread();
|
|
||||||
this.setState({
|
|
||||||
numPages: 1,
|
|
||||||
scrollLocked: false,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
this.scrolledToMarker = true;
|
|
||||||
}
|
|
||||||
this.unreadMarker = ref;
|
|
||||||
}
|
|
||||||
|
|
||||||
dismissUnread() {
|
|
||||||
this.props.api.chat.read(this.props.station);
|
|
||||||
}
|
|
||||||
|
|
||||||
chatWindow(unread) {
|
|
||||||
// Replace with just the "not Firefox" implementation
|
|
||||||
// when Firefox #1042151 is patched.
|
|
||||||
|
|
||||||
const { props, state } = this;
|
|
||||||
|
|
||||||
let messages: IMessage[] = props.envelopes.slice(0);
|
|
||||||
const lastMsgNum = messages.length > 0 ? messages.length : 0;
|
|
||||||
|
|
||||||
if (messages.length > 100 * state.numPages) {
|
|
||||||
messages = messages.slice(0, 100 * state.numPages);
|
|
||||||
}
|
|
||||||
|
|
||||||
const pendingMessages: IMessage[] = (
|
|
||||||
props.pendingMessages.get(props.station) || []
|
|
||||||
).map((value) => ({ ...value, pending: true }));
|
|
||||||
|
|
||||||
if(unread !== 0) {
|
|
||||||
unread += pendingMessages.length;
|
|
||||||
}
|
|
||||||
|
|
||||||
messages = pendingMessages.concat(messages);
|
|
||||||
|
|
||||||
const messageElements = messages.map((msg, i) => {
|
|
||||||
// Render sigil if previous message is not by the same sender
|
|
||||||
const aut = ["author"];
|
|
||||||
const renderSigil =
|
|
||||||
_.get(messages[i + 1], aut) !== _.get(msg, aut, msg.author);
|
|
||||||
const paddingTop = renderSigil;
|
|
||||||
const paddingBot =
|
|
||||||
_.get(messages[i - 1], aut) !== _.get(msg, aut, msg.author);
|
|
||||||
|
|
||||||
const when = ["when"];
|
|
||||||
const dayBreak =
|
|
||||||
moment(_.get(messages[i + 1], when)).format("YYYY.MM.DD") !==
|
|
||||||
moment(_.get(messages[i], when)).format("YYYY.MM.DD");
|
|
||||||
|
|
||||||
const messageElem = (
|
|
||||||
<Message
|
|
||||||
key={msg.uid}
|
|
||||||
msg={msg}
|
|
||||||
contacts={props.contacts}
|
|
||||||
renderSigil={renderSigil}
|
|
||||||
paddingTop={paddingTop}
|
|
||||||
paddingBot={paddingBot}
|
|
||||||
pending={Boolean(msg.pending)}
|
|
||||||
group={props.group}
|
|
||||||
association={props.association}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
if (unread > 0 && i === unread - 1) {
|
|
||||||
return (
|
|
||||||
<Fragment key={msg.uid}>
|
|
||||||
{messageElem}
|
|
||||||
<div
|
|
||||||
ref={this.setUnreadMarker}
|
|
||||||
className="mv2 green2 flex items-center f9"
|
|
||||||
>
|
|
||||||
<hr className="dn-s ma0 w2 b--green2 bt-0" />
|
|
||||||
<p className="mh4">New messages below</p>
|
|
||||||
<hr className="ma0 flex-grow-1 b--green2 bt-0" />
|
|
||||||
{dayBreak && (
|
|
||||||
<p className="gray2 mh4">
|
|
||||||
{moment(_.get(messages[i], when)).calendar()}
|
|
||||||
</p>
|
|
||||||
)}
|
|
||||||
<hr
|
|
||||||
style={{ width: "calc(50% - 48px)" }}
|
|
||||||
className="b--green2 ma0 bt-0"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</Fragment>
|
|
||||||
);
|
|
||||||
} else if (dayBreak) {
|
|
||||||
return (
|
|
||||||
<Fragment key={msg.uid}>
|
|
||||||
{messageElem}
|
|
||||||
<div
|
|
||||||
className="pv3 gray2 b--gray2 flex items-center justify-center f9 "
|
|
||||||
>
|
|
||||||
<p>{moment(_.get(messages[i], when)).calendar()}</p>
|
|
||||||
</div>
|
|
||||||
</Fragment>
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
return messageElem;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
if (navigator.userAgent.includes("Firefox")) {
|
|
||||||
return (
|
|
||||||
<div
|
|
||||||
className="relative overflow-y-scroll h-100"
|
|
||||||
onScroll={this.onScroll}
|
|
||||||
ref={(e) => {
|
|
||||||
this.scrollContainer = e;
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="bg-white bg-gray0-d pt3 pb2 flex flex-column-reverse"
|
|
||||||
style={{ resize: "vertical" }}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
ref={(el) => {
|
|
||||||
this.scrollElement = el;
|
|
||||||
}}
|
|
||||||
></div>
|
|
||||||
{props.chatInitialized && !(props.station in props.inbox) && (
|
|
||||||
<BacklogElement />
|
|
||||||
)}
|
|
||||||
{props.chatSynced &&
|
|
||||||
!(props.station in props.chatSynced) &&
|
|
||||||
messages.length > 0 ? (
|
|
||||||
<ResubscribeElement
|
|
||||||
api={props.api}
|
|
||||||
host={props.match.params.ship}
|
|
||||||
station={props.station}
|
|
||||||
/>
|
|
||||||
) : (
|
|
||||||
<div />
|
|
||||||
)}
|
|
||||||
{messageElements}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
return (
|
|
||||||
<div
|
|
||||||
className="overflow-y-scroll bg-white bg-gray0-d pt3 pb2 flex flex-column-reverse relative"
|
|
||||||
style={{ height: "100%", resize: "vertical" }}
|
|
||||||
onScroll={this.onScroll}
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
ref={(el) => {
|
|
||||||
this.scrollElement = el;
|
|
||||||
}}
|
|
||||||
></div>
|
|
||||||
{props.chatInitialized && !(props.station in props.inbox) && (
|
|
||||||
<BacklogElement />
|
|
||||||
)}
|
|
||||||
{props.chatSynced &&
|
|
||||||
!(props.station in props.chatSynced) &&
|
|
||||||
messages.length > 0 ? (
|
|
||||||
<ResubscribeElement
|
|
||||||
api={props.api}
|
|
||||||
host={props.match.params.ship}
|
|
||||||
station={props.station}
|
|
||||||
/>
|
|
||||||
) : (
|
|
||||||
<div />
|
|
||||||
)}
|
|
||||||
{messageElements}
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
const { props, state } = this;
|
const { props, state } = this;
|
||||||
|
|
||||||
const messages = props.envelopes.slice(0);
|
const lastMsgNum = props.envelopes.length > 0 ? props.envelopes.length : 0;
|
||||||
|
|
||||||
const lastMsgNum = messages.length > 0 ? messages.length : 0;
|
|
||||||
|
|
||||||
const group = Array.from(props.group.members);
|
|
||||||
|
|
||||||
const isinPopout = props.popout ? "popout/" : "";
|
|
||||||
|
|
||||||
const ownerContact =
|
const ownerContact =
|
||||||
window.ship in props.contacts ? props.contacts[window.ship] : false;
|
window.ship in props.contacts ? props.contacts[window.ship] : false;
|
||||||
|
|
||||||
let title = props.station.substr(1);
|
const pendingMessages = (props.pendingMessages.get(props.station) || [])
|
||||||
|
.map((value) => ({
|
||||||
|
...value,
|
||||||
|
pending: true
|
||||||
|
}));
|
||||||
|
|
||||||
if (props.association && "metadata" in props.association) {
|
const isChatMissing =
|
||||||
title =
|
props.chatInitialized &&
|
||||||
props.association.metadata.title !== ""
|
!(props.station in props.inbox) &&
|
||||||
? props.association.metadata.title
|
props.chatSynced &&
|
||||||
: props.station.substr(1);
|
!(props.station in props.chatSynced);
|
||||||
}
|
|
||||||
|
|
||||||
const unread = props.length - state.read;
|
const isChatLoading =
|
||||||
|
props.chatInitialized &&
|
||||||
|
!(props.station in props.inbox) &&
|
||||||
|
props.chatSynced &&
|
||||||
|
(props.station in props.chatSynced);
|
||||||
|
|
||||||
const unreadMsg = unread > 0 && messages[unread - 1];
|
const isChatUnsynced =
|
||||||
|
props.chatSynced &&
|
||||||
|
!(props.station in props.chatSynced) &&
|
||||||
|
props.envelopes.length > 0;
|
||||||
|
|
||||||
const showUnreadNotice =
|
const unreadCount = props.length - props.read;
|
||||||
props.length !== props.read && props.read === state.read;
|
const unreadMsg = unreadCount > 0 && props.envelopes[unreadCount - 1];
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
key={props.station}
|
key={props.station}
|
||||||
className="h-100 w-100 overflow-hidden flex flex-column relative"
|
className="h-100 w-100 overflow-hidden flex flex-column relative">
|
||||||
>
|
<ChatHeader
|
||||||
<div
|
match={props.match}
|
||||||
className="w-100 dn-m dn-l dn-xl inter pt4 pb6 pl3 f8"
|
location={props.location}
|
||||||
style={{ height: "1rem" }}
|
api={props.api}
|
||||||
>
|
group={props.group}
|
||||||
<Link to="/~chat/">{"⟵ All Chats"}</Link>
|
association={props.association}
|
||||||
</div>
|
station={props.station}
|
||||||
|
sidebarShown={props.sidebarShown}
|
||||||
<div
|
popout={props.popout} />
|
||||||
className={
|
<ChatWindow
|
||||||
"pl4 pt2 bb b--gray4 b--gray1-d bg-gray0-d flex relative " +
|
history={props.history}
|
||||||
"overflow-x-auto overflow-y-hidden flex-shrink-0 "
|
isChatMissing={isChatMissing}
|
||||||
}
|
isChatLoading={isChatLoading}
|
||||||
style={{ height: 48 }}
|
isChatUnsynced={isChatUnsynced}
|
||||||
>
|
unreadCount={unreadCount}
|
||||||
<SidebarSwitcher
|
unreadMsg={unreadMsg}
|
||||||
sidebarShown={props.sidebarShown}
|
pendingMessages={pendingMessages}
|
||||||
popout={props.popout}
|
messages={props.envelopes}
|
||||||
api={props.api}
|
length={props.length}
|
||||||
/>
|
contacts={props.contacts}
|
||||||
<Link
|
association={props.association}
|
||||||
to={"/~chat/" + isinPopout + "room" + props.station}
|
group={props.group}
|
||||||
className="pt2 white-d"
|
ship={props.match.params.ship}
|
||||||
>
|
station={props.station}
|
||||||
<h2
|
api={props.api} />
|
||||||
className={
|
|
||||||
"dib f9 fw4 lh-solid v-top " +
|
|
||||||
(title === props.station.substr(1) ? "mono" : "")
|
|
||||||
}
|
|
||||||
style={{ width: "max-content" }}
|
|
||||||
>
|
|
||||||
{title}
|
|
||||||
</h2>
|
|
||||||
</Link>
|
|
||||||
<ChatTabBar
|
|
||||||
{...props}
|
|
||||||
station={props.station}
|
|
||||||
numPeers={group.length}
|
|
||||||
isOwner={deSig(props.match.params.ship) === window.ship}
|
|
||||||
popout={props.popout}
|
|
||||||
api={props.api}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
{!!unreadMsg && showUnreadNotice && (
|
|
||||||
<UnreadNotice
|
|
||||||
unread={unread}
|
|
||||||
unreadMsg={unreadMsg}
|
|
||||||
onRead={() => this.dismissUnread()}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
{this.chatWindow(unread)}
|
|
||||||
<ChatInput
|
<ChatInput
|
||||||
api={props.api}
|
api={props.api}
|
||||||
numMsgs={lastMsgNum}
|
numMsgs={lastMsgNum}
|
||||||
@ -595,13 +135,15 @@ export class ChatScreen extends Component<ChatScreenProps, ChatScreenState> {
|
|||||||
ownerContact={ownerContact}
|
ownerContact={ownerContact}
|
||||||
envelopes={props.envelopes}
|
envelopes={props.envelopes}
|
||||||
contacts={props.contacts}
|
contacts={props.contacts}
|
||||||
onEnter={() => this.setState({ scrollLocked: false })}
|
|
||||||
onUnmount={(msg: string) => this.setState({
|
onUnmount={(msg: string) => this.setState({
|
||||||
messages: this.state.messages.set(props.station, msg)
|
messages: this.state.messages.set(props.station, msg)
|
||||||
})}
|
})}
|
||||||
s3={props.s3}
|
s3={props.s3}
|
||||||
placeholder="Message..."
|
placeholder="Message..."
|
||||||
message={this.state.messages.get(props.station) || ""}
|
message={this.state.messages.get(props.station) || ""}
|
||||||
|
deleteMessage={() => this.setState({
|
||||||
|
messages: this.state.messages.set(props.station, "")
|
||||||
|
})}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
@ -1,21 +1,22 @@
|
|||||||
import React, { Component } from 'react';
|
import React, { Component } from 'react';
|
||||||
|
|
||||||
export class BacklogElement extends Component {
|
export const BacklogElement = (props) => {
|
||||||
render() {
|
if (!props.isChatLoading) {
|
||||||
return (
|
return null;
|
||||||
<div className="center mw6">
|
|
||||||
<div className="db pa3 ma3 ba b--gray4 bg-gray5 b--gray2-d bg-gray1-d white-d flex items-center">
|
|
||||||
<img className="invert-d spin-active v-mid"
|
|
||||||
src="/~chat/img/Spinner.png"
|
|
||||||
width={16}
|
|
||||||
height={16}
|
|
||||||
/>
|
|
||||||
<p className="lh-copy db ml3">
|
|
||||||
Past messages are being restored
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
return (
|
||||||
|
<div className="center mw6">
|
||||||
|
<div className={
|
||||||
|
"db pa3 ma3 ba b--gray4 bg-gray5 b--gray2-d bg-gray1-d " +
|
||||||
|
"white-d flex items-center"
|
||||||
|
}>
|
||||||
|
<img className="invert-d spin-active v-mid"
|
||||||
|
src="/~chat/img/Spinner.png"
|
||||||
|
width={16}
|
||||||
|
height={16}
|
||||||
|
/>
|
||||||
|
<p className="lh-copy db ml3">Past messages are being restored</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
141
pkg/interface/src/apps/chat/components/lib/chat-editor.js
Normal file
141
pkg/interface/src/apps/chat/components/lib/chat-editor.js
Normal file
@ -0,0 +1,141 @@
|
|||||||
|
import React, { Component } from 'react';
|
||||||
|
import { UnControlled as CodeEditor } from 'react-codemirror2';
|
||||||
|
import CodeMirror from 'codemirror';
|
||||||
|
|
||||||
|
import 'codemirror/mode/markdown/markdown';
|
||||||
|
import 'codemirror/addon/display/placeholder';
|
||||||
|
|
||||||
|
import 'codemirror/lib/codemirror.css';
|
||||||
|
|
||||||
|
const BROWSER_REGEX =
|
||||||
|
new RegExp(String(/Android|webOS|iPhone|iPad|iPod|BlackBerry/i));
|
||||||
|
|
||||||
|
|
||||||
|
const MARKDOWN_CONFIG = {
|
||||||
|
name: 'markdown',
|
||||||
|
tokenTypeOverrides: {
|
||||||
|
header: 'presentation',
|
||||||
|
quote: 'presentation',
|
||||||
|
list1: 'presentation',
|
||||||
|
list2: 'presentation',
|
||||||
|
list3: 'presentation',
|
||||||
|
hr: 'presentation',
|
||||||
|
image: 'presentation',
|
||||||
|
imageAltText: 'presentation',
|
||||||
|
imageMarker: 'presentation',
|
||||||
|
formatting: 'presentation',
|
||||||
|
linkInline: 'presentation',
|
||||||
|
linkEmail: 'presentation',
|
||||||
|
linkText: 'presentation',
|
||||||
|
linkHref: 'presentation'
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export default class ChatEditor extends Component {
|
||||||
|
constructor(props) {
|
||||||
|
super(props);
|
||||||
|
|
||||||
|
this.state = {
|
||||||
|
message: props.message
|
||||||
|
};
|
||||||
|
this.editor = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
componentWillUnmount() {
|
||||||
|
this.props.onUnmount(this.state.message);
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidUpdate(prevProps) {
|
||||||
|
const { props } = this;
|
||||||
|
|
||||||
|
if (prevProps.message !== props.message) {
|
||||||
|
this.editor.setValue(props.message);
|
||||||
|
this.editor.setOption('mode', MARKDOWN_CONFIG);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!props.inCodeMode) {
|
||||||
|
this.editor.setOption('mode', MARKDOWN_CONFIG);
|
||||||
|
this.editor.setOption('placeholder', this.props.placeholder);
|
||||||
|
} else {
|
||||||
|
this.editor.setOption('mode', null);
|
||||||
|
this.editor.setOption('placeholder', 'Code...');
|
||||||
|
}
|
||||||
|
const value = this.editor.getValue();
|
||||||
|
|
||||||
|
// Force redraw of placeholder
|
||||||
|
if(value.length === 0) {
|
||||||
|
this.editor.setValue(' ');
|
||||||
|
this.editor.setValue('');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
submit() {
|
||||||
|
if(!this.editor) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let editorMessage = this.editor.getValue();
|
||||||
|
if (editorMessage === '') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.setState({ message: '' });
|
||||||
|
this.props.submit(editorMessage);
|
||||||
|
this.editor.setValue('');
|
||||||
|
}
|
||||||
|
|
||||||
|
messageChange(editor, data, value) {
|
||||||
|
if (this.state.message !== '' && value == '') {
|
||||||
|
this.setState({
|
||||||
|
message: value
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (value == this.props.message || value == '' || value == ' ') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.setState({
|
||||||
|
message: value
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
const { props } = this;
|
||||||
|
|
||||||
|
const codeTheme = props.inCodeMode ? ' code' : '';
|
||||||
|
|
||||||
|
const options = {
|
||||||
|
mode: MARKDOWN_CONFIG,
|
||||||
|
theme: 'tlon' + codeTheme,
|
||||||
|
lineNumbers: false,
|
||||||
|
lineWrapping: true,
|
||||||
|
scrollbarStyle: 'native',
|
||||||
|
cursorHeight: 0.85,
|
||||||
|
placeholder: props.inCodeMode ? 'Code...' : props.placeholder,
|
||||||
|
extraKeys: {
|
||||||
|
'Enter': () => {
|
||||||
|
this.submit();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className="chat fr h-100 flex bg-gray0-d lh-copy pl2 w-100 items-center"
|
||||||
|
style={{ flexGrow: 1, maxHeight: '224px', width: 'calc(100% - 72px)' }}
|
||||||
|
>
|
||||||
|
<CodeEditor
|
||||||
|
value={props.message}
|
||||||
|
options={options}
|
||||||
|
onChange={(e, d, v) => this.messageChange(e, d, v)}
|
||||||
|
editorDidMount={(editor) => {
|
||||||
|
this.editor = editor;
|
||||||
|
if (!(BROWSER_REGEX.test(navigator.userAgent))) {
|
||||||
|
editor.focus();
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
58
pkg/interface/src/apps/chat/components/lib/chat-header.js
Normal file
58
pkg/interface/src/apps/chat/components/lib/chat-header.js
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
import React, { Component, Fragment } from "react";
|
||||||
|
import { Link } from "react-router-dom";
|
||||||
|
|
||||||
|
import { ChatTabBar } from "./chat-tabbar";
|
||||||
|
import { SidebarSwitcher } from "../../../../components/SidebarSwitch";
|
||||||
|
import { deSig } from "../../../../lib/util";
|
||||||
|
|
||||||
|
|
||||||
|
export const ChatHeader = (props) => {
|
||||||
|
const isInPopout = props.popout ? "popout/" : "";
|
||||||
|
const group = Array.from(props.group.members);
|
||||||
|
let title = props.station.substr(1);
|
||||||
|
if (props.association &&
|
||||||
|
"metadata" in props.association &&
|
||||||
|
props.association.metadata.tile !== "") {
|
||||||
|
title = props.association.metadata.title
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Fragment>
|
||||||
|
<div
|
||||||
|
className="w-100 dn-m dn-l dn-xl inter pt4 pb6 pl3 f8"
|
||||||
|
style={{ height: "1rem" }}>
|
||||||
|
<Link to="/~chat/">{"⟵ All Chats"}</Link>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
className={
|
||||||
|
"pl4 pt2 bb b--gray4 b--gray1-d bg-gray0-d flex relative " +
|
||||||
|
"overflow-x-auto overflow-y-hidden flex-shrink-0 "
|
||||||
|
}
|
||||||
|
style={{ height: 48 }}>
|
||||||
|
<SidebarSwitcher
|
||||||
|
sidebarShown={props.sidebarShown}
|
||||||
|
popout={props.popout}
|
||||||
|
api={props.api}
|
||||||
|
/>
|
||||||
|
<Link
|
||||||
|
to={"/~chat/" + isInPopout + "room" + props.station}
|
||||||
|
className="pt2 white-d">
|
||||||
|
<h2
|
||||||
|
className={
|
||||||
|
"dib f9 fw4 lh-solid v-top " +
|
||||||
|
(title === props.station.substr(1) ? "mono" : "")
|
||||||
|
}
|
||||||
|
style={{ width: "max-content" }}>
|
||||||
|
{title}
|
||||||
|
</h2>
|
||||||
|
</Link>
|
||||||
|
<ChatTabBar
|
||||||
|
location={props.location}
|
||||||
|
station={props.station}
|
||||||
|
isOwner={deSig(props.match.params.ship) === window.ship}
|
||||||
|
popout={props.popout}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</Fragment>
|
||||||
|
);
|
||||||
|
}
|
@ -1,143 +1,41 @@
|
|||||||
import React, { Component } from 'react';
|
import React, { Component } from 'react';
|
||||||
import _ from 'lodash';
|
|
||||||
import moment from 'moment';
|
|
||||||
import { UnControlled as CodeEditor } from 'react-codemirror2';
|
|
||||||
import CodeMirror from 'codemirror';
|
|
||||||
|
|
||||||
import 'codemirror/mode/markdown/markdown';
|
|
||||||
import 'codemirror/addon/display/placeholder';
|
|
||||||
|
|
||||||
import 'codemirror/lib/codemirror.css';
|
|
||||||
|
|
||||||
import { Sigil } from '../../../../lib/sigil';
|
import { Sigil } from '../../../../lib/sigil';
|
||||||
import { ShipSearch } from './ship-search';
|
import ChatEditor from './chat-editor';
|
||||||
import { S3Upload } from './s3-upload';
|
import { S3Upload } from './s3-upload';
|
||||||
|
|
||||||
import { uxToHex } from '../../../../lib/util';
|
import { uxToHex } from '../../../../lib/util';
|
||||||
|
|
||||||
const MARKDOWN_CONFIG = {
|
|
||||||
name: 'markdown',
|
const URL_REGEX = new RegExp(String(/^((\w+:\/\/)[-a-zA-Z0-9:@;?&=\/%\+\.\*!'\(\),\$_\{\}\^~\[\]`#|]+)/.source));
|
||||||
tokenTypeOverrides: {
|
|
||||||
header: 'presentation',
|
|
||||||
quote: 'presentation',
|
|
||||||
list1: 'presentation',
|
|
||||||
list2: 'presentation',
|
|
||||||
list3: 'presentation',
|
|
||||||
hr: 'presentation',
|
|
||||||
image: 'presentation',
|
|
||||||
imageAltText: 'presentation',
|
|
||||||
imageMarker: 'presentation',
|
|
||||||
formatting: 'presentation',
|
|
||||||
linkInline: 'presentation',
|
|
||||||
linkEmail: 'presentation',
|
|
||||||
linkText: 'presentation',
|
|
||||||
linkHref: 'presentation'
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
export class ChatInput extends Component {
|
export class ChatInput extends Component {
|
||||||
constructor(props) {
|
constructor(props) {
|
||||||
super(props);
|
super(props);
|
||||||
|
|
||||||
this.state = {
|
this.state = {
|
||||||
message: props.message,
|
inCodeMode: false,
|
||||||
patpSearch: null
|
|
||||||
};
|
};
|
||||||
|
|
||||||
this.textareaRef = React.createRef();
|
this.submit = this.submit.bind(this);
|
||||||
|
|
||||||
this.messageSubmit = this.messageSubmit.bind(this);
|
|
||||||
this.messageChange = this.messageChange.bind(this);
|
|
||||||
|
|
||||||
this.patpAutocomplete = this.patpAutocomplete.bind(this);
|
|
||||||
this.completePatp = this.completePatp.bind(this);
|
|
||||||
this.clearSearch = this.clearSearch.bind(this);
|
|
||||||
|
|
||||||
this.toggleCode = this.toggleCode.bind(this);
|
this.toggleCode = this.toggleCode.bind(this);
|
||||||
|
|
||||||
this.editor = null;
|
|
||||||
|
|
||||||
moment.updateLocale('en', {
|
|
||||||
relativeTime : {
|
|
||||||
past: function(input) {
|
|
||||||
return input === 'just now'
|
|
||||||
? input
|
|
||||||
: input + ' ago';
|
|
||||||
},
|
|
||||||
s : 'just now',
|
|
||||||
future: 'in %s',
|
|
||||||
ss : '%d sec',
|
|
||||||
m: 'a minute',
|
|
||||||
mm: '%d min',
|
|
||||||
h: 'an hr',
|
|
||||||
hh: '%d hrs',
|
|
||||||
d: 'a day',
|
|
||||||
dd: '%d days',
|
|
||||||
M: 'a month',
|
|
||||||
MM: '%d months',
|
|
||||||
y: 'a year',
|
|
||||||
yy: '%d years'
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
componentWillUnmount() {
|
uploadSuccess(url) {
|
||||||
this.props.onUnmount(this.state.message);
|
const { props } = this;
|
||||||
|
props.api.chat.message(
|
||||||
|
props.station,
|
||||||
|
`~${window.ship}`,
|
||||||
|
Date.now(),
|
||||||
|
{ url }
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
nextAutocompleteSuggestion(backward = false) {
|
uploadError(error) {
|
||||||
const { patpSuggestions } = this.state;
|
// no-op for now
|
||||||
let idx = patpSuggestions.findIndex(s => s === this.state.selectedSuggestion);
|
|
||||||
|
|
||||||
idx = backward ? idx - 1 : idx + 1;
|
|
||||||
idx = idx % patpSuggestions.length;
|
|
||||||
if(idx < 0) {
|
|
||||||
idx = patpSuggestions.length - 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.setState({ selectedSuggestion: patpSuggestions[idx] });
|
|
||||||
}
|
}
|
||||||
|
|
||||||
patpAutocomplete(message) {
|
toggleCode() {
|
||||||
const match = /~([a-zA-Z\-]*)$/.exec(message);
|
|
||||||
|
|
||||||
if (!match ) {
|
|
||||||
this.setState({ patpSearch: null });
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
this.setState({ patpSearch: match[1].toLowerCase() });
|
|
||||||
}
|
|
||||||
|
|
||||||
clearSearch() {
|
|
||||||
this.setState({
|
this.setState({
|
||||||
patpSearch: null
|
inCodeMode: !this.state.inCodeMode
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
completePatp(suggestion) {
|
|
||||||
if(!this.editor) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const newMessage = this.editor.getValue().replace(
|
|
||||||
/[a-zA-Z\-]*$/,
|
|
||||||
suggestion
|
|
||||||
);
|
|
||||||
this.editor.setValue(newMessage);
|
|
||||||
const lastRow = this.editor.lastLine();
|
|
||||||
const lastCol = this.editor.getLineHandle(lastRow).text.length;
|
|
||||||
this.editor.setCursor(lastRow, lastCol);
|
|
||||||
this.setState({
|
|
||||||
patpSearch: null
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
messageChange(editor, data, value) {
|
|
||||||
const { patpSearch } = this.state;
|
|
||||||
if(patpSearch !== null) {
|
|
||||||
this.patpAutocomplete(value, false);
|
|
||||||
}
|
|
||||||
this.setState({
|
|
||||||
message: value
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -154,7 +52,7 @@ export class ChatInput extends Component {
|
|||||||
me: letter
|
me: letter
|
||||||
};
|
};
|
||||||
} else if (this.isUrl(letter)) {
|
} else if (this.isUrl(letter)) {
|
||||||
return {
|
return {
|
||||||
url: letter
|
url: letter
|
||||||
};
|
};
|
||||||
} else {
|
} else {
|
||||||
@ -166,41 +64,40 @@ export class ChatInput extends Component {
|
|||||||
|
|
||||||
isUrl(string) {
|
isUrl(string) {
|
||||||
try {
|
try {
|
||||||
const websiteTest = new RegExp(String(/^((\w+:\/\/)[-a-zA-Z0-9:@;?&=\/%\+\.\*!'\(\),\$_\{\}\^~\[\]`#|]+)/.source)
|
return URL_REGEX.test(string);
|
||||||
);
|
|
||||||
return websiteTest.test(string);
|
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
messageSubmit() {
|
submit(text) {
|
||||||
if(!this.editor) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const { props, state } = this;
|
const { props, state } = this;
|
||||||
const editorMessage = this.editor.getValue();
|
if (state.inCodeMode) {
|
||||||
|
this.setState({
|
||||||
if (editorMessage === '') {
|
inCodeMode: false
|
||||||
return;
|
}, () => {
|
||||||
}
|
props.api.chat.message(
|
||||||
|
props.station,
|
||||||
props.onEnter();
|
`~${window.ship}`,
|
||||||
|
Date.now(), {
|
||||||
if(state.code) {
|
code: {
|
||||||
props.api.chat.message(props.station, `~${window.ship}`, Date.now(), {
|
expression: text,
|
||||||
code: {
|
output: undefined
|
||||||
expression: editorMessage,
|
}
|
||||||
output: undefined
|
}
|
||||||
}
|
);
|
||||||
});
|
});
|
||||||
this.editor.setValue('');
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let messages = [];
|
||||||
let message = [];
|
let message = [];
|
||||||
let isInCodeBlock = false;
|
let isInCodeBlock = false;
|
||||||
let endOfCodeBlock = false;
|
let endOfCodeBlock = false;
|
||||||
editorMessage.split(/\r?\n/).forEach((line) => {
|
text.split(/\r?\n/).forEach((line, index) => {
|
||||||
|
if (index !== 0) {
|
||||||
|
message.push('\n');
|
||||||
|
}
|
||||||
// A line of backticks enters and exits a codeblock
|
// A line of backticks enters and exits a codeblock
|
||||||
if (line.startsWith('```')) {
|
if (line.startsWith('```')) {
|
||||||
// But we need to check if we've ended a codeblock
|
// But we need to check if we've ended a codeblock
|
||||||
@ -209,10 +106,9 @@ export class ChatInput extends Component {
|
|||||||
} else {
|
} else {
|
||||||
endOfCodeBlock = false;
|
endOfCodeBlock = false;
|
||||||
}
|
}
|
||||||
if (isInCodeBlock) {
|
|
||||||
message.push(`\n${line}`);
|
if (isInCodeBlock || endOfCodeBlock) {
|
||||||
} else if (endOfCodeBlock) {
|
message.push(line);
|
||||||
message.push(`\n${line}\n`);
|
|
||||||
} else {
|
} else {
|
||||||
line.split(/\s/).forEach((str) => {
|
line.split(/\s/).forEach((str) => {
|
||||||
if (
|
if (
|
||||||
@ -226,46 +122,41 @@ export class ChatInput extends Component {
|
|||||||
) {
|
) {
|
||||||
isInCodeBlock = false;
|
isInCodeBlock = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.isUrl(str) && !isInCodeBlock) {
|
if (this.isUrl(str) && !isInCodeBlock) {
|
||||||
if (message.length > 0) {
|
if (message.length > 0) {
|
||||||
message = message.join(' ');
|
// If we're in the middle of a message, add it to the stack and reset
|
||||||
message = this.getLetterType(message);
|
messages.push(message);
|
||||||
props.api.chat.message(
|
|
||||||
props.station,
|
|
||||||
`~${window.ship}`,
|
|
||||||
Date.now(),
|
|
||||||
message
|
|
||||||
);
|
|
||||||
message = [];
|
message = [];
|
||||||
}
|
}
|
||||||
const URL = this.getLetterType(str);
|
messages.push([str]);
|
||||||
props.api.chat.message(
|
message = [];
|
||||||
props.station,
|
|
||||||
`~${window.ship}`,
|
|
||||||
Date.now(),
|
|
||||||
URL
|
|
||||||
);
|
|
||||||
} else {
|
} else {
|
||||||
message.push(str);
|
message.push(str);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
if (message.length > 0) {
|
if (message.length) {
|
||||||
message = message.join(' ');
|
// Add any remaining message
|
||||||
message = this.getLetterType(message);
|
messages.push(message);
|
||||||
props.api.chat.message(
|
|
||||||
props.station,
|
|
||||||
`~${window.ship}`,
|
|
||||||
Date.now(),
|
|
||||||
message
|
|
||||||
);
|
|
||||||
message = [];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
props.deleteMessage();
|
||||||
|
|
||||||
|
messages.forEach((message) => {
|
||||||
|
if (message.length > 0) {
|
||||||
|
message = this.getLetterType(message.join(' '));
|
||||||
|
props.api.chat.message(
|
||||||
|
props.station,
|
||||||
|
`~${window.ship}`,
|
||||||
|
Date.now(),
|
||||||
|
message
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
// perf testing:
|
// perf testing:
|
||||||
/*let closure = () => {
|
/*let closure = () => {
|
||||||
let x = 0;
|
let x = 0;
|
||||||
@ -284,27 +175,6 @@ export class ChatInput extends Component {
|
|||||||
};
|
};
|
||||||
this.closure = closure.bind(this);
|
this.closure = closure.bind(this);
|
||||||
setTimeout(this.closure, 2000);*/
|
setTimeout(this.closure, 2000);*/
|
||||||
|
|
||||||
this.editor.setValue('');
|
|
||||||
}
|
|
||||||
|
|
||||||
toggleCode() {
|
|
||||||
if(this.state.code) {
|
|
||||||
this.setState({ code: false });
|
|
||||||
this.editor.setOption('mode', MARKDOWN_CONFIG);
|
|
||||||
this.editor.setOption('placeholder', this.props.placeholder);
|
|
||||||
} else {
|
|
||||||
this.setState({ code: true });
|
|
||||||
this.editor.setOption('mode', null);
|
|
||||||
this.editor.setOption('placeholder', 'Code...');
|
|
||||||
}
|
|
||||||
const value = this.editor.getValue();
|
|
||||||
|
|
||||||
// Force redraw of placeholder
|
|
||||||
if(value.length === 0) {
|
|
||||||
this.editor.setValue(' ');
|
|
||||||
this.editor.setValue('');
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
uploadSuccess(url) {
|
uploadSuccess(url) {
|
||||||
@ -330,7 +200,7 @@ export class ChatInput extends Component {
|
|||||||
const sigilClass = props.ownerContact
|
const sigilClass = props.ownerContact
|
||||||
? '' : 'mix-blend-diff';
|
? '' : 'mix-blend-diff';
|
||||||
|
|
||||||
const img = (props.ownerContact && (props.ownerContact.avatar !== null))
|
const avatar = (props.ownerContact && (props.ownerContact.avatar !== null))
|
||||||
? <img src={props.ownerContact.avatar} height={24} width={24} className="dib" />
|
? <img src={props.ownerContact.avatar} height={24} width={24} className="dib" />
|
||||||
: <Sigil
|
: <Sigil
|
||||||
ship={window.ship}
|
ship={window.ship}
|
||||||
@ -339,82 +209,33 @@ export class ChatInput extends Component {
|
|||||||
classes={sigilClass}
|
classes={sigilClass}
|
||||||
/>;
|
/>;
|
||||||
|
|
||||||
const candidates = _.chain(this.props.envelopes)
|
|
||||||
.defaultTo([])
|
|
||||||
.map('author')
|
|
||||||
.uniq()
|
|
||||||
.reverse()
|
|
||||||
.value();
|
|
||||||
|
|
||||||
const codeTheme = state.code ? ' code' : '';
|
|
||||||
|
|
||||||
const options = {
|
|
||||||
mode: MARKDOWN_CONFIG,
|
|
||||||
theme: 'tlon' + codeTheme,
|
|
||||||
lineNumbers: false,
|
|
||||||
lineWrapping: true,
|
|
||||||
scrollbarStyle: 'native',
|
|
||||||
cursorHeight: 0.85,
|
|
||||||
placeholder: state.code ? 'Code...' : props.placeholder,
|
|
||||||
extraKeys: {
|
|
||||||
Tab: cm =>
|
|
||||||
this.patpAutocomplete(cm.getValue(), true),
|
|
||||||
'Enter': () => {
|
|
||||||
this.messageSubmit();
|
|
||||||
if (this.state.code) {
|
|
||||||
this.toggleCode();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
'Shift-3': cm =>
|
|
||||||
cm.getValue().length === 0
|
|
||||||
? this.toggleCode()
|
|
||||||
: CodeMirror.Pass
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="chat pa3 cf flex black white-d bt b--gray4 b--gray1-d bg-white bg-gray0-d relative"
|
<div className={
|
||||||
style={{ flexGrow: 1 }}
|
"pa3 cf flex black white-d bt b--gray4 b--gray1-d bg-white " +
|
||||||
>
|
"bg-gray0-d relative"
|
||||||
<ShipSearch
|
}
|
||||||
popover
|
style={{ flexGrow: 1 }}>
|
||||||
onSelect={this.completePatp}
|
<div className="fl"
|
||||||
onClear={this.clearSearch}
|
style={{
|
||||||
contacts={props.contacts}
|
marginTop: 6,
|
||||||
candidates={candidates}
|
flexBasis: 24,
|
||||||
searchTerm={this.state.patpSearch}
|
height: 24
|
||||||
cm={this.editor}
|
}}>
|
||||||
/>
|
{avatar}
|
||||||
<div
|
|
||||||
className="fl"
|
|
||||||
style={{
|
|
||||||
marginTop: 6,
|
|
||||||
flexBasis: 24,
|
|
||||||
height: 24
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{img}
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="fr h-100 flex bg-gray0-d lh-copy pl2 w-100 items-center"
|
|
||||||
style={{ flexGrow: 1, maxHeight: '224px', width: 'calc(100% - 72px)' }}
|
|
||||||
>
|
|
||||||
<CodeEditor
|
|
||||||
value={this.props.message}
|
|
||||||
options={options}
|
|
||||||
editorDidMount={(editor) => {
|
|
||||||
this.editor = editor;
|
|
||||||
if (!/Android|webOS|iPhone|iPad|iPod|BlackBerry/i.test(
|
|
||||||
navigator.userAgent
|
|
||||||
)) {
|
|
||||||
editor.focus();
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
onChange={(e, d, v) => this.messageChange(e, d, v)}
|
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
|
<ChatEditor
|
||||||
|
inCodeMode={state.inCodeMode}
|
||||||
|
submit={this.submit}
|
||||||
|
onUnmount={props.onUnmount}
|
||||||
|
message={props.message}
|
||||||
|
placeholder='Message...' />
|
||||||
<div className="ml2 mr2"
|
<div className="ml2 mr2"
|
||||||
style={{ height: '16px', width: '16px', flexBasis: 16, marginTop: 10 }}>
|
style={{
|
||||||
|
height: '16px',
|
||||||
|
width: '16px',
|
||||||
|
flexBasis: 16,
|
||||||
|
marginTop: 10
|
||||||
|
}}>
|
||||||
<S3Upload
|
<S3Upload
|
||||||
configuration={props.s3.configuration}
|
configuration={props.s3.configuration}
|
||||||
credentials={props.s3.credentials}
|
credentials={props.s3.credentials}
|
||||||
@ -422,13 +243,20 @@ export class ChatInput extends Component {
|
|||||||
uploadError={this.uploadError.bind(this)}
|
uploadError={this.uploadError.bind(this)}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<div style={{ height: '16px', width: '16px', flexBasis: 16, marginTop: 10 }}>
|
<div style={{
|
||||||
<img
|
height: '16px',
|
||||||
style={{ filter: state.code && 'invert(100%)', height: '14px', width: '14px' }}
|
width: '16px',
|
||||||
onClick={this.toggleCode}
|
flexBasis: 16,
|
||||||
src="/~chat/img/CodeEval.png"
|
marginTop: 10
|
||||||
className="contrast-10-d bg-white bg-none-d ba b--gray1-d br1"
|
}}>
|
||||||
/>
|
<img style={{
|
||||||
|
filter: state.inCodeMode && 'invert(100%)',
|
||||||
|
height: '14px',
|
||||||
|
width: '14px',
|
||||||
|
}}
|
||||||
|
onClick={this.toggleCode}
|
||||||
|
src="/~chat/img/CodeEval.png"
|
||||||
|
className="contrast-10-d bg-white bg-none-d ba b--gray1-d br1" />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
84
pkg/interface/src/apps/chat/components/lib/chat-message.tsx
Normal file
84
pkg/interface/src/apps/chat/components/lib/chat-message.tsx
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
import React, { PureComponent, Fragment } from "react";
|
||||||
|
import moment from "moment";
|
||||||
|
|
||||||
|
import { Message } from "./message";
|
||||||
|
|
||||||
|
type IMessage = Envelope & { pending?: boolean };
|
||||||
|
|
||||||
|
|
||||||
|
export const ChatMessage = (props) => {
|
||||||
|
const {
|
||||||
|
msg,
|
||||||
|
previousMsg,
|
||||||
|
nextMsg,
|
||||||
|
isLastUnread,
|
||||||
|
group,
|
||||||
|
association,
|
||||||
|
contacts,
|
||||||
|
unreadRef
|
||||||
|
} = props;
|
||||||
|
|
||||||
|
// Render sigil if previous message is not by the same sender
|
||||||
|
const aut = ["author"];
|
||||||
|
const renderSigil =
|
||||||
|
_.get(nextMsg, aut) !== _.get(msg, aut, msg.author);
|
||||||
|
const paddingTop = renderSigil;
|
||||||
|
const paddingBot =
|
||||||
|
_.get(previousMsg, aut) !== _.get(msg, aut, msg.author);
|
||||||
|
|
||||||
|
const when = ["when"];
|
||||||
|
const dayBreak =
|
||||||
|
moment(_.get(nextMsg, when)).format("YYYY.MM.DD") !==
|
||||||
|
moment(_.get(msg, when)).format("YYYY.MM.DD");
|
||||||
|
|
||||||
|
const messageElem = (
|
||||||
|
<Message
|
||||||
|
key={msg.uid}
|
||||||
|
msg={msg}
|
||||||
|
renderSigil={renderSigil}
|
||||||
|
paddingTop={paddingTop}
|
||||||
|
paddingBot={paddingBot}
|
||||||
|
pending={Boolean(msg.pending)}
|
||||||
|
group={group}
|
||||||
|
contacts={contacts}
|
||||||
|
association={association}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
if (props.isLastUnread) {
|
||||||
|
return (
|
||||||
|
<Fragment key={msg.uid}>
|
||||||
|
{messageElem}
|
||||||
|
<div ref={unreadRef}
|
||||||
|
className="mv2 green2 flex items-center f9">
|
||||||
|
<hr className="dn-s ma0 w2 b--green2 bt-0" />
|
||||||
|
<p className="mh4">New messages below</p>
|
||||||
|
<hr className="ma0 flex-grow-1 b--green2 bt-0" />
|
||||||
|
{dayBreak && (
|
||||||
|
<p className="gray2 mh4">
|
||||||
|
{moment(_.get(msg, when)).calendar()}
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
|
<hr
|
||||||
|
style={{ width: "calc(50% - 48px)" }}
|
||||||
|
className="b--green2 ma0 bt-0"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</Fragment>
|
||||||
|
);
|
||||||
|
} else if (dayBreak) {
|
||||||
|
return (
|
||||||
|
<Fragment key={msg.uid}>
|
||||||
|
{messageElem}
|
||||||
|
<div
|
||||||
|
className="pv3 gray2 b--gray2 flex items-center justify-center f9 "
|
||||||
|
>
|
||||||
|
<p>{moment(_.get(msg, when)).calendar()}</p>
|
||||||
|
</div>
|
||||||
|
</Fragment>
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return messageElem;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
@ -0,0 +1,143 @@
|
|||||||
|
import React, { Component, Fragment } from "react";
|
||||||
|
|
||||||
|
import { scrollIsAtTop, scrollIsAtBottom } from "../../../../lib/util";
|
||||||
|
|
||||||
|
// Restore chat position on FF when new messages come in
|
||||||
|
const recalculateScrollTop = (lastScrollHeight, scrollContainer) => {
|
||||||
|
if (!scrollContainer || !lastScrollHeight) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const newScrollTop = scrollContainer.scrollHeight - lastScrollHeight;
|
||||||
|
if (scrollContainer.scrollTop !== 0 ||
|
||||||
|
scrollContainer.scrollTop === newScrollTop) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
scrollContainer.scrollTop = scrollContainer.scrollHeight - lastScrollHeight;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
export class ChatScrollContainer extends Component {
|
||||||
|
constructor(props) {
|
||||||
|
super(props);
|
||||||
|
|
||||||
|
// only for FF
|
||||||
|
this.state = {
|
||||||
|
lastScrollHeight: null
|
||||||
|
};
|
||||||
|
|
||||||
|
this.isTriggeredScroll = false;
|
||||||
|
|
||||||
|
this.isAtBottom = true;
|
||||||
|
this.isAtTop = false;
|
||||||
|
|
||||||
|
this.containerDidScroll = this.containerDidScroll.bind(this);
|
||||||
|
|
||||||
|
this.containerRef = React.createRef();
|
||||||
|
this.scrollRef = React.createRef();
|
||||||
|
}
|
||||||
|
|
||||||
|
containerDidScroll(e) {
|
||||||
|
const { props } = this;
|
||||||
|
if (scrollIsAtTop(e.target)) {
|
||||||
|
// Save scroll position for FF
|
||||||
|
if (navigator.userAgent.includes("Firefox")) {
|
||||||
|
this.setState({
|
||||||
|
lastScrollHeight: e.target.scrollHeight,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!this.isAtTop) {
|
||||||
|
props.scrollIsAtTop();
|
||||||
|
}
|
||||||
|
|
||||||
|
this.isTriggeredScroll = false;
|
||||||
|
this.isAtBottom = false;
|
||||||
|
this.isAtTop = true;
|
||||||
|
} else if (scrollIsAtBottom(e.target) && !this.isTriggeredScroll) {
|
||||||
|
if (!this.isAtBottom) {
|
||||||
|
props.scrollIsAtBottom();
|
||||||
|
}
|
||||||
|
|
||||||
|
this.isTriggeredScroll = false;
|
||||||
|
this.isAtBottom = true;
|
||||||
|
this.isAtTop = false;
|
||||||
|
} else {
|
||||||
|
this.isAtBottom = false;
|
||||||
|
this.isAtTop = false;
|
||||||
|
this.isTriggeredScroll = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
// Replace with just the "not Firefox" implementation
|
||||||
|
// when Firefox #1042151 is patched.
|
||||||
|
|
||||||
|
if (navigator.userAgent.includes("Firefox")) {
|
||||||
|
return this.firefoxScrollContainer();
|
||||||
|
} else {
|
||||||
|
return this.normalScrollContainer();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
firefoxScrollContainer() {
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className="relative overflow-y-scroll h-100"
|
||||||
|
onScroll={this.containerDidScroll}
|
||||||
|
ref={this.containerRef}>
|
||||||
|
<div
|
||||||
|
className="bg-white bg-gray0-d pt3 pb2 flex flex-column-reverse"
|
||||||
|
style={{ resize: "vertical" }}>
|
||||||
|
<div ref={this.scrollRef}></div>
|
||||||
|
{this.props.children}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
normalScrollContainer() {
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className={
|
||||||
|
"overflow-y-scroll bg-white bg-gray0-d pt3 pb2 flex " +
|
||||||
|
"flex-column-reverse relative"
|
||||||
|
}
|
||||||
|
style={{ height: "100%", resize: "vertical" }}
|
||||||
|
onScroll={this.containerDidScroll}>
|
||||||
|
<div ref={this.scrollRef}></div>
|
||||||
|
{this.props.children}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
scrollToBottom() {
|
||||||
|
this.isTriggeredScroll = true;
|
||||||
|
if (this.scrollRef.current) {
|
||||||
|
this.scrollRef.current.scrollIntoView(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (navigator.userAgent.includes("Firefox")) {
|
||||||
|
recalculateScrollTop(
|
||||||
|
this.state.lastScrollHeight,
|
||||||
|
this.scrollContainer
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
scrollToReference(ref) {
|
||||||
|
this.isTriggeredScroll = true;
|
||||||
|
if (this.scrollRef.current && ref.current) {
|
||||||
|
ref.current.scrollIntoView({ block: 'center' });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (navigator.userAgent.includes("Firefox")) {
|
||||||
|
recalculateScrollTop(
|
||||||
|
this.state.lastScrollHeight,
|
||||||
|
this.scrollContainer
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -1,66 +1,41 @@
|
|||||||
import React, { Component } from 'react';
|
import React, { Component } from 'react';
|
||||||
import { Link } from 'react-router-dom';
|
import { Link } from 'react-router-dom';
|
||||||
|
|
||||||
export class ChatTabBar extends Component {
|
export const ChatTabBar = (props) => {
|
||||||
render() {
|
const {
|
||||||
const props = this.props;
|
location,
|
||||||
|
station
|
||||||
|
} = props;
|
||||||
|
let setColor = '', popout = '';
|
||||||
|
|
||||||
let memColor = '',
|
if (location.pathname.includes('/settings')) {
|
||||||
setColor = '',
|
setColor = 'black white-d';
|
||||||
popout = '';
|
} else {
|
||||||
|
setColor = 'gray3';
|
||||||
if (props.location.pathname.includes('/settings')) {
|
|
||||||
memColor = 'gray3';
|
|
||||||
setColor = 'black white-d';
|
|
||||||
} else if (props.location.pathname.includes('/members')) {
|
|
||||||
memColor = 'black white-d';
|
|
||||||
setColor = 'gray3';
|
|
||||||
} else {
|
|
||||||
memColor = 'gray3';
|
|
||||||
setColor = 'gray3';
|
|
||||||
}
|
|
||||||
|
|
||||||
popout = props.location.pathname.includes('/popout')
|
|
||||||
? 'popout/' : '';
|
|
||||||
|
|
||||||
const hidePopoutIcon = (this.props.popout)
|
|
||||||
? 'dn-m dn-l dn-xl' : 'dib-m dib-l dib-xl';
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div className="dib flex-shrink-0 flex-grow-1">
|
|
||||||
{props.isOwner ? (
|
|
||||||
<div className={'dib pt2 f9 pl6 lh-solid'}>
|
|
||||||
<Link
|
|
||||||
className={'no-underline ' + memColor}
|
|
||||||
to={'/~chat/' + popout + 'members' + props.station}
|
|
||||||
>
|
|
||||||
Members
|
|
||||||
</Link>
|
|
||||||
</div>
|
|
||||||
) : (
|
|
||||||
<div className="dib" style={{ width: 0 }}></div>
|
|
||||||
)}
|
|
||||||
<div className={'dib pt2 f9 pl6 pr6 lh-solid'}>
|
|
||||||
<Link
|
|
||||||
className={'no-underline ' + setColor}
|
|
||||||
to={'/~chat/' + popout + 'settings' + props.station}
|
|
||||||
>
|
|
||||||
Settings
|
|
||||||
</Link>
|
|
||||||
</div>
|
|
||||||
<a href={'/~chat/popout/room' + props.station} rel="noopener noreferrer"
|
|
||||||
target="_blank"
|
|
||||||
className="dib fr pr1"
|
|
||||||
style={{ paddingTop: '8px' }}
|
|
||||||
>
|
|
||||||
<img
|
|
||||||
className={'flex-shrink-0 pr3 dn ' + hidePopoutIcon}
|
|
||||||
src="/~chat/img/popout.png"
|
|
||||||
height="16"
|
|
||||||
width="16"
|
|
||||||
/>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const hidePopoutIcon = (popout)
|
||||||
|
? 'dn-m dn-l dn-xl' : 'dib-m dib-l dib-xl';
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="dib flex-shrink-0 flex-grow-1">
|
||||||
|
<div className={'dib pt2 f9 pl6 pr6 lh-solid'}>
|
||||||
|
<Link
|
||||||
|
className={'no-underline ' + setColor}
|
||||||
|
to={'/~chat/' + popout + 'settings' + station}>
|
||||||
|
Settings
|
||||||
|
</Link>
|
||||||
|
</div>
|
||||||
|
<a href={'/~chat/popout/room' + station} rel="noopener noreferrer"
|
||||||
|
target="_blank"
|
||||||
|
className="dib fr pr1"
|
||||||
|
style={{ paddingTop: '8px' }}>
|
||||||
|
<img
|
||||||
|
className={'flex-shrink-0 pr3 dn ' + hidePopoutIcon}
|
||||||
|
src="/~chat/img/popout.png"
|
||||||
|
height="16"
|
||||||
|
width="16" />
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
194
pkg/interface/src/apps/chat/components/lib/chat-window.tsx
Normal file
194
pkg/interface/src/apps/chat/components/lib/chat-window.tsx
Normal file
@ -0,0 +1,194 @@
|
|||||||
|
import React, { Component, Fragment } from "react";
|
||||||
|
|
||||||
|
import { ChatMessage } from './chat-message';
|
||||||
|
import { ChatScrollContainer } from "./chat-scroll-container";
|
||||||
|
import { UnreadNotice } from "./unread-notice";
|
||||||
|
import { ResubscribeElement } from "./resubscribe-element";
|
||||||
|
import { BacklogElement } from "./backlog-element";
|
||||||
|
|
||||||
|
const MAX_BACKLOG_SIZE = 1000;
|
||||||
|
const DEFAULT_BACKLOG_SIZE = 200;
|
||||||
|
const PAGE_SIZE = 50;
|
||||||
|
const INITIAL_LOAD = 20;
|
||||||
|
|
||||||
|
|
||||||
|
export class ChatWindow extends Component {
|
||||||
|
constructor(props) {
|
||||||
|
super(props);
|
||||||
|
this.state = {
|
||||||
|
numPages: 1,
|
||||||
|
};
|
||||||
|
|
||||||
|
this.hasAskedForMessages = false;
|
||||||
|
|
||||||
|
this.dismissUnread = this.dismissUnread.bind(this);
|
||||||
|
this.scrollIsAtBottom = this.scrollIsAtBottom.bind(this);
|
||||||
|
this.scrollIsAtTop = this.scrollIsAtTop.bind(this);
|
||||||
|
|
||||||
|
this.scrollReference = React.createRef();
|
||||||
|
this.unreadReference = React.createRef();
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidMount() {
|
||||||
|
this.initialFetch();
|
||||||
|
|
||||||
|
if (this.state.numPages === 1 && this.props.unreadCount < INITIAL_LOAD) {
|
||||||
|
this.dismissUnread();
|
||||||
|
this.scrollToBottom();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
initialFetch() {
|
||||||
|
const { props } = this;
|
||||||
|
if (props.messages.length > 0) {
|
||||||
|
const unreadUnloaded = props.unreadCount - props.messages.length;
|
||||||
|
|
||||||
|
if (unreadUnloaded <= MAX_BACKLOG_SIZE &&
|
||||||
|
unreadUnloaded + INITIAL_LOAD > DEFAULT_BACKLOG_SIZE) {
|
||||||
|
this.fetchBacklog(unreadUnloaded + INITIAL_LOAD);
|
||||||
|
} else {
|
||||||
|
this.fetchBacklog(DEFAULT_BACKLOG_SIZE);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
setTimeout(() => {
|
||||||
|
this.initialFetch();
|
||||||
|
}, 2000);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidUpdate(prevProps, prevState) {
|
||||||
|
const { props, state } = this;
|
||||||
|
|
||||||
|
if (props.isChatMissing) {
|
||||||
|
props.history.push("/~chat");
|
||||||
|
} else if (props.messages.length >= prevProps.messages.length + 10) {
|
||||||
|
this.hasAskedForMessages = false;
|
||||||
|
let numPages = props.unreadCount > 0 ?
|
||||||
|
Math.ceil(props.unreadCount / PAGE_SIZE) : this.state.numPages;
|
||||||
|
|
||||||
|
if (this.state.numPages === numPages) {
|
||||||
|
if (props.unreadCount > 20) {
|
||||||
|
this.scrollToUnread();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
this.setState({ numPages }, () => {
|
||||||
|
if (props.unreadCount > 20) {
|
||||||
|
this.scrollToUnread();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} else if (
|
||||||
|
state.numPages === 1 &&
|
||||||
|
this.props.unreadCount < INITIAL_LOAD &&
|
||||||
|
this.props.unreadCount > 0
|
||||||
|
) {
|
||||||
|
this.dismissUnread();
|
||||||
|
this.scrollToBottom();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
scrollIsAtTop() {
|
||||||
|
const { props, state } = this;
|
||||||
|
this.setState({ numPages: state.numPages + 1 }, () => {
|
||||||
|
if (state.numPages * PAGE_SIZE < props.length) {
|
||||||
|
this.fetchBacklog(DEFAULT_BACKLOG_SIZE);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
scrollIsAtBottom() {
|
||||||
|
if (this.state.numPages !== 1) {
|
||||||
|
this.setState({ numPages: 1 });
|
||||||
|
this.dismissUnread();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
scrollToBottom() {
|
||||||
|
if (this.scrollReference.current) {
|
||||||
|
this.scrollReference.current.scrollToBottom();
|
||||||
|
}
|
||||||
|
if (this.state.numPages !== 1) {
|
||||||
|
this.setState({ numPages: 1 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
scrollToUnread() {
|
||||||
|
if (this.scrollReference.current && this.unreadReference.current) {
|
||||||
|
this.scrollReference.current.scrollToReference(this.unreadReference);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
dismissUnread() {
|
||||||
|
this.props.api.chat.read(this.props.station);
|
||||||
|
}
|
||||||
|
|
||||||
|
fetchBacklog(size) {
|
||||||
|
const { props } = this;
|
||||||
|
|
||||||
|
if (
|
||||||
|
props.messages.length >= props.length ||
|
||||||
|
this.hasAskedForMessages ||
|
||||||
|
props.length <= 0
|
||||||
|
) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const start =
|
||||||
|
props.length - props.messages[props.messages.length - 1].number;
|
||||||
|
if (start > 0) {
|
||||||
|
const end = start + size < props.length ? start + size : props.length;
|
||||||
|
props.api.chat.fetchMessages(start + 1, end, props.station);
|
||||||
|
this.hasAskedForMessages = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
const { props, state } = this;
|
||||||
|
const sliceLength = Math.min(
|
||||||
|
state.numPages * PAGE_SIZE,
|
||||||
|
props.messages.length + props.pendingMessages.length
|
||||||
|
);
|
||||||
|
const messages =
|
||||||
|
props.pendingMessages
|
||||||
|
.concat(props.messages)
|
||||||
|
.slice(0, sliceLength);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Fragment>
|
||||||
|
<UnreadNotice
|
||||||
|
unreadCount={props.unreadCount}
|
||||||
|
unreadMsg={props.unreadMsg}
|
||||||
|
dismissUnread={this.dismissUnread} />
|
||||||
|
<ChatScrollContainer
|
||||||
|
ref={this.scrollReference}
|
||||||
|
scrollIsAtBottom={this.scrollIsAtBottom}
|
||||||
|
scrollIsAtTop={this.scrollIsAtTop}>
|
||||||
|
<BacklogElement isChatLoading={props.isChatLoading} />
|
||||||
|
<ResubscribeElement
|
||||||
|
api={props.api}
|
||||||
|
host={props.ship}
|
||||||
|
station={props.station}
|
||||||
|
isChatUnsynced={props.isChatUnsynced}
|
||||||
|
/>
|
||||||
|
{ messages.map((msg, i) => (
|
||||||
|
<ChatMessage
|
||||||
|
unreadRef={this.unreadReference}
|
||||||
|
isLastUnread={
|
||||||
|
props.unreadCount > 0 &&
|
||||||
|
i === props.unreadCount - 1 &&
|
||||||
|
state.numPages !== 1
|
||||||
|
}
|
||||||
|
msg={msg}
|
||||||
|
previousMsg={messages[i - 1]}
|
||||||
|
nextMsg={messages[i + 1]}
|
||||||
|
association={props.association}
|
||||||
|
group={props.group}
|
||||||
|
contacts={props.contacts} />
|
||||||
|
))
|
||||||
|
}
|
||||||
|
</ChatScrollContainer>
|
||||||
|
</Fragment>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -1,8 +1,8 @@
|
|||||||
import React, { Component } from 'react';
|
import React, { Component } from 'react';
|
||||||
|
import { Link } from 'react-router-dom';
|
||||||
import ReactMarkdown from 'react-markdown';
|
import ReactMarkdown from 'react-markdown';
|
||||||
import RemarkDisableTokenizers from 'remark-disable-tokenizers';
|
import RemarkDisableTokenizers from 'remark-disable-tokenizers';
|
||||||
import urbitOb from 'urbit-ob';
|
import urbitOb from 'urbit-ob';
|
||||||
import { Link } from 'react-router-dom';
|
|
||||||
|
|
||||||
const DISABLED_BLOCK_TOKENS = [
|
const DISABLED_BLOCK_TOKENS = [
|
||||||
'indentedCode',
|
'indentedCode',
|
||||||
|
@ -1,8 +1,6 @@
|
|||||||
import React, { Component } from 'react';
|
import React, { Component } from 'react';
|
||||||
|
|
||||||
|
const IMAGE_REGEX = new RegExp(/(jpg|img|png|gif|tiff|jpeg|webp|webm|svg)$/i);
|
||||||
const IMAGE_REGEX =
|
|
||||||
/(jpg|img|png|gif|tiff|jpeg|JPG|IMG|PNG|TIFF|GIF|webp|WEBP|webm|WEBM|svg|SVG)$/;
|
|
||||||
|
|
||||||
const YOUTUBE_REGEX =
|
const YOUTUBE_REGEX =
|
||||||
new RegExp(
|
new RegExp(
|
||||||
@ -25,8 +23,7 @@ export default class UrlContent extends Component {
|
|||||||
let unfoldState = this.state.unfold;
|
let unfoldState = this.state.unfold;
|
||||||
unfoldState = !unfoldState;
|
unfoldState = !unfoldState;
|
||||||
this.setState({ unfold: unfoldState });
|
this.setState({ unfold: unfoldState });
|
||||||
const iframe = this.refs.iframe;
|
this.iframe.setAttribute('src', this.iframe.dataset.src);
|
||||||
iframe.setAttribute('src', iframe.getAttribute('data-src'));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
@ -42,13 +39,12 @@ export default class UrlContent extends Component {
|
|||||||
className="o-80-d"
|
className="o-80-d"
|
||||||
src={content.url}
|
src={content.url}
|
||||||
style={{
|
style={{
|
||||||
width: '50%',
|
maxWidth: '18rem'
|
||||||
maxWidth: '250px'
|
|
||||||
}}
|
}}
|
||||||
></img>
|
></img>
|
||||||
);
|
);
|
||||||
return (
|
return (
|
||||||
<a className={`f7 lh-copy v-top word-break-all`}
|
<a className='f7 lh-copy v-top word-break-all'
|
||||||
href={content.url}
|
href={content.url}
|
||||||
target="_blank"
|
target="_blank"
|
||||||
rel="noopener noreferrer"
|
rel="noopener noreferrer"
|
||||||
@ -63,7 +59,9 @@ export default class UrlContent extends Component {
|
|||||||
? 'db' : 'dn')}
|
? 'db' : 'dn')}
|
||||||
>
|
>
|
||||||
<iframe
|
<iframe
|
||||||
ref="iframe"
|
ref={(el) => {
|
||||||
|
this.iframe = el;
|
||||||
|
}}
|
||||||
width="560"
|
width="560"
|
||||||
height="315"
|
height="315"
|
||||||
data-src={`https://www.youtube.com/embed/${ytMatch[1]}`}
|
data-src={`https://www.youtube.com/embed/${ytMatch[1]}`}
|
||||||
@ -75,12 +73,15 @@ export default class UrlContent extends Component {
|
|||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
<a href={content.url}
|
<a href={content.url}
|
||||||
className={`f7 lh-copy v-top bb b--white-d word-break-all`}
|
className='f7 lh-copy v-top bb b--white-d word-break-all'
|
||||||
href={content.url}
|
|
||||||
target="_blank"
|
target="_blank"
|
||||||
rel="noopener noreferrer">{content.url}</a>
|
rel="noopener noreferrer"
|
||||||
<a className="ml2 f7 pointer lh-copy v-top"
|
>
|
||||||
onClick={e => this.unfoldEmbed()}>
|
{content.url}
|
||||||
|
</a>
|
||||||
|
<a className="bs ml2 f7 pointer lh-copy v-top"
|
||||||
|
onClick={e => this.unfoldEmbed()}
|
||||||
|
>
|
||||||
[embed]
|
[embed]
|
||||||
</a>
|
</a>
|
||||||
{contents}
|
{contents}
|
||||||
@ -88,7 +89,7 @@ export default class UrlContent extends Component {
|
|||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
return (
|
return (
|
||||||
<a className={`f7 lh-copy v-top bb b--white-d b--black word-break-all`}
|
<a className='f7 lh-copy v-top bb b--white-d b--black word-break-all'
|
||||||
href={content.url}
|
href={content.url}
|
||||||
target="_blank"
|
target="_blank"
|
||||||
rel="noopener noreferrer"
|
rel="noopener noreferrer"
|
||||||
|
51
pkg/interface/src/apps/chat/components/lib/delete-button.js
Normal file
51
pkg/interface/src/apps/chat/components/lib/delete-button.js
Normal file
@ -0,0 +1,51 @@
|
|||||||
|
import React, { Component } from 'react';
|
||||||
|
|
||||||
|
|
||||||
|
export const DeleteButton = (props) => {
|
||||||
|
const { isOwner, station, changeLoading, api } = props;
|
||||||
|
const leaveButtonClasses = (!isOwner) ? 'pointer' : 'c-default';
|
||||||
|
const deleteButtonClasses = (isOwner) ?
|
||||||
|
'b--red2 red2 pointer bg-gray0-d' :
|
||||||
|
'b--gray3 gray3 bg-gray0-d c-default';
|
||||||
|
|
||||||
|
const deleteChat = () => {
|
||||||
|
changeLoading(
|
||||||
|
true,
|
||||||
|
true,
|
||||||
|
isOwner ? 'Deleting chat...' : 'Leaving chat...',
|
||||||
|
() => {
|
||||||
|
api.chat.delete(station);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="w-100 cf">
|
||||||
|
<div className={'w-100 fl mt3 ' + ((isOwner) ? 'o-30' : '')}>
|
||||||
|
<p className="f8 mt3 lh-copy db">Leave Chat</p>
|
||||||
|
<p className="f9 gray2 db mb4">
|
||||||
|
Remove this chat from your chat list.{' '}
|
||||||
|
You will need to request for access again.
|
||||||
|
</p>
|
||||||
|
<a onClick={(!isOwner) ? deleteChat : null}
|
||||||
|
className={
|
||||||
|
'dib f9 black gray4-d bg-gray0-d ba pa2 b--black b--gray1-d ' +
|
||||||
|
leaveButtonClasses
|
||||||
|
}>
|
||||||
|
Leave this chat
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div className={'w-100 fl mt3 ' + ((!isOwner) ? 'o-30' : '')}>
|
||||||
|
<p className="f8 mt3 lh-copy db">Delete Chat</p>
|
||||||
|
<p className="f9 gray2 db mb4">
|
||||||
|
Permanently delete this chat.{' '}
|
||||||
|
All current members will no longer see this chat.
|
||||||
|
</p>
|
||||||
|
<a onClick={(isOwner) ? deleteChat : null}
|
||||||
|
className={'dib f9 ba pa2 ' + deleteButtonClasses}
|
||||||
|
>Delete this chat</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
104
pkg/interface/src/apps/chat/components/lib/groupify-button.js
Normal file
104
pkg/interface/src/apps/chat/components/lib/groupify-button.js
Normal file
@ -0,0 +1,104 @@
|
|||||||
|
import React, { Component } from 'react';
|
||||||
|
import Toggle from '../../../../components/toggle';
|
||||||
|
import { InviteSearch } from '../../../../components/InviteSearch';
|
||||||
|
|
||||||
|
|
||||||
|
export class GroupifyButton extends Component {
|
||||||
|
|
||||||
|
constructor(props) {
|
||||||
|
super(props);
|
||||||
|
|
||||||
|
this.state = {
|
||||||
|
inclusive: false,
|
||||||
|
targetGroup: null
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
changeTargetGroup(target) {
|
||||||
|
if (target.groups.length === 1) {
|
||||||
|
this.setState({ targetGroup: target.groups[0] });
|
||||||
|
} else {
|
||||||
|
this.setState({ targetGroup: null });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
changeInclusive(event) {
|
||||||
|
this.setState({ inclusive: Boolean(event.target.checked) });
|
||||||
|
}
|
||||||
|
|
||||||
|
renderInclusiveToggle() {
|
||||||
|
return this.state.targetGroup ? (
|
||||||
|
<div className="mt4">
|
||||||
|
<Toggle
|
||||||
|
boolean={inclusive}
|
||||||
|
change={this.changeInclusive.bind(this)}
|
||||||
|
/>
|
||||||
|
<span className="dib f9 white-d inter ml3">
|
||||||
|
Add all members to group
|
||||||
|
</span>
|
||||||
|
<p className="f9 gray2 pt1" style={{ paddingLeft: 40 }}>
|
||||||
|
Add chat members to the group if they aren't in it yet
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
) : <div />;
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
const { inclusive, targetGroup } = this.state;
|
||||||
|
const {
|
||||||
|
api,
|
||||||
|
isOwner,
|
||||||
|
association,
|
||||||
|
associations,
|
||||||
|
contacts,
|
||||||
|
groups,
|
||||||
|
station
|
||||||
|
} = this.props;
|
||||||
|
|
||||||
|
const groupPath = association['group-path'];
|
||||||
|
const ownedUnmanagedVillage =
|
||||||
|
isOwner &&
|
||||||
|
!contacts[groupPath];
|
||||||
|
|
||||||
|
if (!ownedUnmanagedVillage) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={'w-100 fl mt3'} style={{ maxWidth: '29rem' }}>
|
||||||
|
<p className="f8 mt3 lh-copy db">Convert Chat</p>
|
||||||
|
<p className="f9 gray2 db mb4">
|
||||||
|
Convert this chat into a group with associated chat, or select a
|
||||||
|
group to add this chat to.
|
||||||
|
</p>
|
||||||
|
<InviteSearch
|
||||||
|
groups={groups}
|
||||||
|
contacts={contacts}
|
||||||
|
associations={associations}
|
||||||
|
groupResults={true}
|
||||||
|
shipResults={false}
|
||||||
|
invites={{
|
||||||
|
groups: targetGroup ? [targetGroup] : [],
|
||||||
|
ships: []
|
||||||
|
}}
|
||||||
|
setInvite={this.changeTargetGroup.bind(this)}
|
||||||
|
/>
|
||||||
|
{this.renderInclusiveToggle()}
|
||||||
|
<a onClick={() => {
|
||||||
|
changeLoading(true, true, 'Converting to group...', () => {
|
||||||
|
api.chat.groupify(
|
||||||
|
station, targetGroup, inclusive
|
||||||
|
).then(() => {
|
||||||
|
changeLoading(false, false, '', () => {});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}}
|
||||||
|
className={
|
||||||
|
'dib f9 black gray4-d bg-gray0-d ba pa2 mt4 b--black ' +
|
||||||
|
'b--gray1-d pointer'
|
||||||
|
}>Convert to group</a>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -5,49 +5,41 @@ import { uxToHex, cite, writeText } from '../../../../lib/util';
|
|||||||
import moment from 'moment';
|
import moment from 'moment';
|
||||||
|
|
||||||
|
|
||||||
export class Message extends Component {
|
export const Message = (props) => {
|
||||||
constructor() {
|
const pending = props.msg.pending ? ' o-40' : '';
|
||||||
super();
|
const containerClass =
|
||||||
this.state = {
|
props.renderSigil ?
|
||||||
copied: false
|
`w-100 f7 pl3 pt4 pr3 cf flex lh-copy ` + pending :
|
||||||
};
|
'w-100 pr3 cf hide-child flex' + pending;
|
||||||
}
|
|
||||||
|
|
||||||
render() {
|
const timestamp =
|
||||||
const { props, state } = this;
|
moment.unix(props.msg.when / 1000).format(
|
||||||
|
props.renderSigil ? 'hh:mm a' : 'hh:mm'
|
||||||
const pending = props.msg.pending ? ' o-40' : '';
|
|
||||||
const containerClass =
|
|
||||||
props.renderSigil ?
|
|
||||||
`w-100 f7 pl3 pt4 pr3 cf flex lh-copy ` + pending :
|
|
||||||
'w-100 pr3 cf hide-child flex' + pending;
|
|
||||||
|
|
||||||
const timestamp =
|
|
||||||
moment.unix(props.msg.when / 1000).format(
|
|
||||||
props.renderSigil ? 'hh:mm a' : 'hh:mm'
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div
|
|
||||||
ref={this.containerRef}
|
|
||||||
className={containerClass}
|
|
||||||
style={{
|
|
||||||
minHeight: 'min-content'
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{
|
|
||||||
props.renderSigil ? (
|
|
||||||
this.renderWithSigil(timestamp)
|
|
||||||
) : (
|
|
||||||
this.renderWithoutSigil(timestamp)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
</div>
|
|
||||||
);
|
);
|
||||||
}
|
|
||||||
|
|
||||||
renderWithSigil(timestamp) {
|
return (
|
||||||
const { props, state } = this;
|
<div className={containerClass}
|
||||||
|
style={{
|
||||||
|
minHeight: 'min-content'
|
||||||
|
}}>
|
||||||
|
{
|
||||||
|
props.renderSigil ? (
|
||||||
|
renderWithSigil(props, timestamp)
|
||||||
|
) : (
|
||||||
|
<div className="flex w-100">
|
||||||
|
<p className="child pt2 pl2 pr1 mono f9 gray2 dib">{timestamp}</p>
|
||||||
|
<div className="fr f7 clamp-message white-d pr3 lh-copy"
|
||||||
|
style={{ flexGrow: 1 }}>
|
||||||
|
<MessageContent letter={props.msg.letter} />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const renderWithSigil = (props, timestamp) => {
|
||||||
const paddingTop = props.paddingTop ? { 'paddingTop': '6px' } : '';
|
const paddingTop = props.paddingTop ? { 'paddingTop': '6px' } : '';
|
||||||
const datestamp =
|
const datestamp =
|
||||||
'~' + moment.unix(props.msg.when / 1000).format('YYYY.M.D');
|
'~' + moment.unix(props.msg.when / 1000).format('YYYY.M.D');
|
||||||
@ -85,19 +77,15 @@ export class Message extends Component {
|
|||||||
<p className={`v-mid f9 gray2 dib mr3 c-default`}>
|
<p className={`v-mid f9 gray2 dib mr3 c-default`}>
|
||||||
<span
|
<span
|
||||||
className={
|
className={
|
||||||
'mw5 dib truncate pointer ' +
|
'mw5 db truncate pointer ' +
|
||||||
(contact.nickname || state.copied ? '' : 'mono')
|
(contact.nickname ? '' : 'mono')
|
||||||
}
|
}
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
writeText(props.msg.author);
|
writeText(props.msg.author);
|
||||||
this.setState({ copied: true });
|
|
||||||
setTimeout(() => {
|
|
||||||
this.setState({ copied: false });
|
|
||||||
}, 800);
|
|
||||||
}}
|
}}
|
||||||
title={`~${props.msg.author}`}
|
title={`~${props.msg.author}`}
|
||||||
>
|
>
|
||||||
{state.copied && 'Copied' || name}
|
{name}
|
||||||
</span>
|
</span>
|
||||||
</p>
|
</p>
|
||||||
<p className={`v-mid mono f9 gray2 dib`}>{timestamp}</p>
|
<p className={`v-mid mono f9 gray2 dib`}>{timestamp}</p>
|
||||||
@ -111,17 +99,3 @@ export class Message extends Component {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
renderWithoutSigil(timestamp) {
|
|
||||||
const { props } = this;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div className="flex w-100">
|
|
||||||
<p className="child pt2 pl2 pr1 mono f9 gray2 dib">{timestamp}</p>
|
|
||||||
<div className="fr f7 clamp-message white-d pr3 lh-copy"
|
|
||||||
style={{ flexGrow: 1 }}>
|
|
||||||
<MessageContent letter={props.msg.letter} />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
71
pkg/interface/src/apps/chat/components/lib/metadata-color.js
Normal file
71
pkg/interface/src/apps/chat/components/lib/metadata-color.js
Normal file
@ -0,0 +1,71 @@
|
|||||||
|
import React, { Component } from 'react';
|
||||||
|
|
||||||
|
import { uxToHex } from '../../../../lib/util';
|
||||||
|
|
||||||
|
|
||||||
|
export class MetadataColor extends Component {
|
||||||
|
|
||||||
|
constructor(props) {
|
||||||
|
super(props);
|
||||||
|
this.state = {
|
||||||
|
color: props.initialValue
|
||||||
|
};
|
||||||
|
|
||||||
|
this.changeColor = this.changeColor.bind(this);
|
||||||
|
this.submitColor = this.submitColor.bind(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidUpdate(prevProps) {
|
||||||
|
const { props } = this;
|
||||||
|
if (prevProps.initialValue !== props.initialValue) {
|
||||||
|
this.setState({ color: props.initialValue });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
changeColor(event) {
|
||||||
|
this.setState({ color: event.target.value });
|
||||||
|
}
|
||||||
|
|
||||||
|
submitColor() {
|
||||||
|
const { props, state } = this;
|
||||||
|
|
||||||
|
let color = state.color;
|
||||||
|
if (color.startsWith('#')) {
|
||||||
|
color = state.color.substr(1);
|
||||||
|
}
|
||||||
|
const hexExp = /([0-9A-Fa-f]{6})/;
|
||||||
|
const hexTest = hexExp.exec(color);
|
||||||
|
if (!props.isDisabled && hexTest && (state.color !== props.initialValue)) {
|
||||||
|
props.setValue(color);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
const { props, state } = this;
|
||||||
|
return (
|
||||||
|
<div className={'cf w-100 mb3 ' + ((props.isDisabled) ? 'o-30' : '')}>
|
||||||
|
<p className="f8 lh-copy">Change color</p>
|
||||||
|
<p className="f9 gray2 db mb4">Give this chat a color when viewing group channels</p>
|
||||||
|
<div className="relative w-100 flex"
|
||||||
|
style={{ maxWidth: '10rem' }}
|
||||||
|
>
|
||||||
|
<div className="absolute"
|
||||||
|
style={{
|
||||||
|
height: 16,
|
||||||
|
width: 16,
|
||||||
|
backgroundColor: state.color,
|
||||||
|
top: 13,
|
||||||
|
left: 11
|
||||||
|
}} />
|
||||||
|
<input
|
||||||
|
className={'pl7 f8 ba b--gray3 b--gray2-d bg-gray0-d white-d ' +
|
||||||
|
'focus-b--black focus-b--white-d pa3 db w-100 flex-auto mr3'}
|
||||||
|
value={state.color}
|
||||||
|
disabled={props.isDisabled}
|
||||||
|
onChange={this.changeColor}
|
||||||
|
onBlur={this.submitColor} />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
52
pkg/interface/src/apps/chat/components/lib/metadata-input.js
Normal file
52
pkg/interface/src/apps/chat/components/lib/metadata-input.js
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
import React, { Component } from 'react';
|
||||||
|
|
||||||
|
export class MetadataInput extends Component {
|
||||||
|
|
||||||
|
constructor(props) {
|
||||||
|
super(props);
|
||||||
|
|
||||||
|
this.state = {
|
||||||
|
value: props.initialValue
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidUpdate(prevProps) {
|
||||||
|
const { props } = this;
|
||||||
|
if (prevProps.initialValue !== props.initialValue) {
|
||||||
|
this.setState({ value: props.initialValue });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
const {
|
||||||
|
title,
|
||||||
|
description,
|
||||||
|
isDisabled,
|
||||||
|
setValue
|
||||||
|
} = this.props;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={'w-100 mb3 fl ' + ((isDisabled) ? 'o-30' : '')}>
|
||||||
|
<p className="f8 lh-copy">{title}</p>
|
||||||
|
<p className="f9 gray2 db mb4">{description}</p>
|
||||||
|
<div className="relative w-100 flex" style={{ maxWidth: '29rem' }}>
|
||||||
|
<input
|
||||||
|
className={'f8 ba b--gray3 b--gray2-d bg-gray0-d white-d ' +
|
||||||
|
'focus-b--black focus-b--white-d pa3 db w-100 flex-auto mr3'}
|
||||||
|
type="text"
|
||||||
|
value={this.state.value}
|
||||||
|
disabled={isDisabled}
|
||||||
|
onChange={(e) => {
|
||||||
|
this.setState({ value: e.target.value });
|
||||||
|
}}
|
||||||
|
onBlur={() => {
|
||||||
|
if (!isDisabled) {
|
||||||
|
setValue(this.state.value || '');
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,90 @@
|
|||||||
|
import React, { Component } from 'react';
|
||||||
|
|
||||||
|
import { MetadataColor } from './metadata-color';
|
||||||
|
import { MetadataInput } from './metadata-input';
|
||||||
|
import { uxToHex } from '../../../../lib/util';
|
||||||
|
|
||||||
|
|
||||||
|
export const MetadataSettings = (props) => {
|
||||||
|
const {
|
||||||
|
isOwner,
|
||||||
|
association,
|
||||||
|
changeLoading,
|
||||||
|
api,
|
||||||
|
station
|
||||||
|
} = props;
|
||||||
|
|
||||||
|
const title =
|
||||||
|
(props.association && 'metadata' in props.association) ?
|
||||||
|
association.metadata.title : '';
|
||||||
|
const description =
|
||||||
|
(props.association && 'metadata' in props.association) ?
|
||||||
|
association.metadata.description : '';
|
||||||
|
const color =
|
||||||
|
(props.association && 'metadata' in props.association) ?
|
||||||
|
`#${uxToHex(props.association.metadata.color)}` : '';
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="cf mt6">
|
||||||
|
<MetadataInput
|
||||||
|
title='Rename'
|
||||||
|
description='Change the name of this chat'
|
||||||
|
isDisabled={!isOwner}
|
||||||
|
initialValue={title}
|
||||||
|
setValue={(val) => {
|
||||||
|
changeLoading(false, true, 'Editing chat...', () => {
|
||||||
|
api.metadata.metadataAdd(
|
||||||
|
'chat',
|
||||||
|
association['app-path'],
|
||||||
|
association['group-path'],
|
||||||
|
val,
|
||||||
|
association.metadata.description,
|
||||||
|
association.metadata['date-created'],
|
||||||
|
uxToHex(association.metadata.color)
|
||||||
|
).then(() => {
|
||||||
|
changeLoading(false, false, '', () => {});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}} />
|
||||||
|
<MetadataInput
|
||||||
|
title='Change description'
|
||||||
|
description='Change the description of this chat'
|
||||||
|
isDisabled={!isOwner}
|
||||||
|
initialValue={description}
|
||||||
|
setValue={(val) => {
|
||||||
|
changeLoading(false, true, 'Editing chat...', () => {
|
||||||
|
api.metadata.metadataAdd(
|
||||||
|
'chat',
|
||||||
|
association['app-path'],
|
||||||
|
association['group-path'],
|
||||||
|
association.metadata.title,
|
||||||
|
val,
|
||||||
|
association.metadata['date-created'],
|
||||||
|
uxToHex(association.metadata.color)
|
||||||
|
).then(() => {
|
||||||
|
changeLoading(false, false, '', () => {});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}} />
|
||||||
|
<MetadataColor
|
||||||
|
initialValue={color}
|
||||||
|
isDisabled={!isOwner}
|
||||||
|
setValue={(val) => {
|
||||||
|
changeLoading(false, true, 'Editing chat...', () => {
|
||||||
|
props.api.metadata.metadataAdd(
|
||||||
|
'chat',
|
||||||
|
association['app-path'],
|
||||||
|
association['group-path'],
|
||||||
|
association.metadata.title,
|
||||||
|
association.metadata.description,
|
||||||
|
association.metadata['date-created'],
|
||||||
|
val
|
||||||
|
).then(() => {
|
||||||
|
changeLoading(false, false, '', () => {});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}} />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
@ -9,19 +9,24 @@ export class ResubscribeElement extends Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
return (
|
const { props } = this;
|
||||||
<div className="db pa3 ma3 ba b--yellow2 bg-yellow0">
|
if (props.isChatUnsynced) {
|
||||||
<p className="lh-copy db">
|
return (
|
||||||
Your ship has been disconnected from the chat's host.
|
<div className="db pa3 ma3 ba b--yellow2 bg-yellow0">
|
||||||
This may be due to a bad connection, going offline, lack of permission,
|
<p className="lh-copy db">
|
||||||
or an over-the-air update.
|
Your ship has been disconnected from the chat's host.
|
||||||
</p>
|
This may be due to a bad connection, going offline, lack of permission,
|
||||||
<a onClick={this.onClickResubscribe.bind(this)}
|
or an over-the-air update.
|
||||||
className="db underline black pointer mt3"
|
</p>
|
||||||
>
|
<a onClick={this.onClickResubscribe.bind(this)}
|
||||||
Reconnect to this chat
|
className="db underline black pointer mt3"
|
||||||
</a>
|
>
|
||||||
</div>
|
Reconnect to this chat
|
||||||
);
|
</a>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,37 +1,41 @@
|
|||||||
import React, { Component } from 'react';
|
import React, { Component } from 'react';
|
||||||
import moment from 'moment';
|
import moment from 'moment';
|
||||||
|
|
||||||
export class UnreadNotice extends Component {
|
export const UnreadNotice = (props) => {
|
||||||
render() {
|
const { unreadCount, unreadMsg, dismissUnread } = props;
|
||||||
const { unread, unreadMsg, onRead } = this.props;
|
|
||||||
|
|
||||||
let datestamp = moment.unix(unreadMsg.when / 1000).format('YYYY.M.D');
|
if (!unreadMsg || (unreadCount === 0)) {
|
||||||
const timestamp = moment.unix(unreadMsg.when / 1000).format('HH:mm');
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
if (datestamp === moment().format('YYYY.M.D')) {
|
let datestamp = moment.unix(unreadMsg.when / 1000).format('YYYY.M.D');
|
||||||
datestamp = null;
|
const timestamp = moment.unix(unreadMsg.when / 1000).format('HH:mm');
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
if (datestamp === moment().format('YYYY.M.D')) {
|
||||||
<div
|
datestamp = null;
|
||||||
style={{ left: '0px' }}
|
}
|
||||||
className="pa4 w-100 absolute z-1 unread-notice"
|
|
||||||
>
|
return (
|
||||||
<div className="ba b--green2 green2 bg-white bg-gray0-d flex items-center pa2 f9 justify-between br1">
|
<div style={{ left: '0px' }}
|
||||||
<p className="lh-copy db">
|
className="pa4 w-100 absolute z-1 unread-notice">
|
||||||
{unread} new messages since{' '}
|
<div className={
|
||||||
{datestamp && (
|
"ba b--green2 green2 bg-white bg-gray0-d flex items-center " +
|
||||||
<>
|
"pa2 f9 justify-between br1"
|
||||||
<span className="green3">~{datestamp}</span> at{' '}
|
}>
|
||||||
</>
|
<p className="lh-copy db">
|
||||||
)}
|
{unreadCount} new messages since{' '}
|
||||||
<span className="green3">{timestamp}</span>
|
{datestamp && (
|
||||||
</p>
|
<>
|
||||||
<div onClick={onRead} className="ml4 inter b--green2 pointer tr lh-copy">
|
<span className="green3">~{datestamp}</span> at{' '}
|
||||||
Mark as Read
|
</>
|
||||||
</div>
|
)}
|
||||||
|
<span className="green3">{timestamp}</span>
|
||||||
|
</p>
|
||||||
|
<div onClick={dismissUnread}
|
||||||
|
className="ml4 inter b--green2 pointer tr lh-copy">
|
||||||
|
Mark as Read
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
</div>
|
||||||
}
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,94 +0,0 @@
|
|||||||
import React, { Component } from 'react';
|
|
||||||
|
|
||||||
import { Link } from 'react-router-dom';
|
|
||||||
|
|
||||||
import { deSig } from '../../../lib/util';
|
|
||||||
import { ChatTabBar } from './lib/chat-tabbar';
|
|
||||||
import { MemberElement } from './lib/member-element';
|
|
||||||
import { InviteElement } from './lib/invite-element';
|
|
||||||
import { SidebarSwitcher } from '../../../components/SidebarSwitch';
|
|
||||||
import { GroupView } from '../../../components/Group';
|
|
||||||
import { PatpNoSig } from '../../../types/noun';
|
|
||||||
|
|
||||||
export class MemberScreen extends Component {
|
|
||||||
constructor(props) {
|
|
||||||
super(props);
|
|
||||||
this.inviteShips = this.inviteShips.bind(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
inviteShips(ships) {
|
|
||||||
const { props } = this;
|
|
||||||
return props.api.chat.invite(props.station, ships.map(s => `~${s}`));
|
|
||||||
}
|
|
||||||
|
|
||||||
render() {
|
|
||||||
const { props } = this;
|
|
||||||
|
|
||||||
const isinPopout = this.props.popout ? 'popout/' : '';
|
|
||||||
|
|
||||||
let title = props.station.substr(1);
|
|
||||||
|
|
||||||
if (props.association && 'metadata' in props.association) {
|
|
||||||
title =
|
|
||||||
props.association.metadata.title !== ''
|
|
||||||
? props.association.metadata.title
|
|
||||||
: props.station.substr(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div className='h-100 w-100 overflow-x-hidden flex flex-column white-d'>
|
|
||||||
<div
|
|
||||||
className='w-100 dn-m dn-l dn-xl inter pt4 pb6 pl3 f8'
|
|
||||||
style={{ height: '1rem' }}
|
|
||||||
>
|
|
||||||
<Link to='/~chat/'>{'⟵ All Chats'}</Link>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className={`pl4 pt2 bb b--gray4 b--gray1-d bg-gray0-d flex relative
|
|
||||||
overflow-x-scroll overflow-x-auto-l overflow-x-auto-xl flex-shrink-0`}
|
|
||||||
style={{ height: 48 }}
|
|
||||||
>
|
|
||||||
<SidebarSwitcher
|
|
||||||
sidebarShown={this.props.sidebarShown}
|
|
||||||
popout={this.props.popout}
|
|
||||||
api={this.props.api}
|
|
||||||
/>
|
|
||||||
<Link
|
|
||||||
to={'/~chat/' + isinPopout + 'room' + props.station}
|
|
||||||
className='pt2 white-d'
|
|
||||||
>
|
|
||||||
<h2
|
|
||||||
className={
|
|
||||||
'dib f9 fw4 lh-solid v-top ' +
|
|
||||||
(title === props.station.substr(1) ? 'mono' : '')
|
|
||||||
}
|
|
||||||
style={{ width: 'max-content' }}
|
|
||||||
>
|
|
||||||
{title}
|
|
||||||
</h2>
|
|
||||||
</Link>
|
|
||||||
<ChatTabBar
|
|
||||||
{...props}
|
|
||||||
station={props.station}
|
|
||||||
numPeers={5}
|
|
||||||
isOwner={deSig(props.match.params.ship) === window.ship}
|
|
||||||
popout={this.props.popout}
|
|
||||||
api={props.api}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div className='w-100 pl3 mt0 mt4-m mt4-l mt4-xl cf pr6'>
|
|
||||||
{ props.association['group-path'] && (
|
|
||||||
<GroupView
|
|
||||||
permissions
|
|
||||||
group={props.group}
|
|
||||||
resourcePath={props.association['group-path'] || ''}
|
|
||||||
associations={props.associations}
|
|
||||||
groups={props.groups}
|
|
||||||
inviteShips={this.inviteShips}
|
|
||||||
contacts={props.contacts}
|
|
||||||
/> )}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,12 +1,15 @@
|
|||||||
import React, { Component } from 'react';
|
import React, { Component, Fragment } from 'react';
|
||||||
import { deSig, uxToHex, writeText } from '../../../lib/util';
|
import { deSig } from '../../../lib/util';
|
||||||
import { Link } from 'react-router-dom';
|
import { Link } from 'react-router-dom';
|
||||||
|
|
||||||
|
import { ChatHeader } from './lib/chat-header';
|
||||||
|
import { MetadataSettings } from './lib/metadata-settings';
|
||||||
|
import { DeleteButton } from './lib/delete-button';
|
||||||
|
import { GroupifyButton } from './lib/groupify-button';
|
||||||
import { Spinner } from '../../../components/Spinner';
|
import { Spinner } from '../../../components/Spinner';
|
||||||
import { ChatTabBar } from './lib/chat-tabbar';
|
import { ChatTabBar } from './lib/chat-tabbar';
|
||||||
import { InviteSearch } from '../../../components/InviteSearch';
|
|
||||||
import SidebarSwitcher from '../../../components/SidebarSwitch';
|
import SidebarSwitcher from '../../../components/SidebarSwitch';
|
||||||
import Toggle from '../../../components/toggle';
|
|
||||||
|
|
||||||
export class SettingsScreen extends Component {
|
export class SettingsScreen extends Component {
|
||||||
constructor(props) {
|
constructor(props) {
|
||||||
@ -14,444 +17,127 @@ export class SettingsScreen extends Component {
|
|||||||
|
|
||||||
this.state = {
|
this.state = {
|
||||||
isLoading: false,
|
isLoading: false,
|
||||||
title: '',
|
|
||||||
description: '',
|
|
||||||
color: '',
|
|
||||||
// groupify settings
|
|
||||||
targetGroup: null,
|
|
||||||
inclusive: false,
|
|
||||||
awaiting: false,
|
awaiting: false,
|
||||||
type: 'Editing chat...'
|
type: 'Editing chat...'
|
||||||
};
|
};
|
||||||
|
|
||||||
this.renderDelete = this.renderDelete.bind(this);
|
this.changeLoading = this.changeLoading.bind(this);
|
||||||
this.changeTargetGroup = this.changeTargetGroup.bind(this);
|
|
||||||
this.changeInclusive = this.changeInclusive.bind(this);
|
|
||||||
this.changeTitle = this.changeTitle.bind(this);
|
|
||||||
this.changeDescription = this.changeDescription.bind(this);
|
|
||||||
this.changeColor = this.changeColor.bind(this);
|
|
||||||
this.submitColor = this.submitColor.bind(this);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
componentDidMount() {
|
componentDidMount() {
|
||||||
const { props } = this;
|
if (this.state.isLoading && (this.props.station in this.props.inbox)) {
|
||||||
if (props.association && 'metadata' in props.association) {
|
this.setState({ isLoading: false });
|
||||||
this.setState({
|
|
||||||
title: props.association.metadata.title,
|
|
||||||
description: props.association.metadata.description,
|
|
||||||
color: `#${uxToHex(props.association.metadata.color)}`
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
componentDidUpdate(prevProps) {
|
componentDidUpdate(prevProps) {
|
||||||
const { props, state } = this;
|
const { props, state } = this;
|
||||||
if (Boolean(state.isLoading) && !(props.station in props.inbox)) {
|
if (state.isLoading && !(props.station in props.inbox)) {
|
||||||
this.setState({
|
this.setState({
|
||||||
isLoading: false
|
isLoading: false
|
||||||
}, () => {
|
}, () => {
|
||||||
props.history.push('/~chat');
|
props.history.push('/~chat');
|
||||||
});
|
});
|
||||||
}
|
} else if (state.isLoading && (props.station in props.inbox)) {
|
||||||
|
this.setState({ isLoading: false });
|
||||||
if ((state.title === '') && (prevProps !== props)) {
|
|
||||||
if (props.association && 'metadata' in props.association)
|
|
||||||
this.setState({
|
|
||||||
title: props.association.metadata.title,
|
|
||||||
description: props.association.metadata.description,
|
|
||||||
color: `#${uxToHex(props.association.metadata.color)}`
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
changeTargetGroup(target) {
|
changeLoading(isLoading, awaiting, type, closure) {
|
||||||
if (target.groups.length === 1) {
|
|
||||||
this.setState({ targetGroup: target.groups[0] });
|
|
||||||
} else {
|
|
||||||
this.setState({ targetGroup: null });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
changeInclusive(event) {
|
|
||||||
this.setState({ inclusive: Boolean(event.target.checked) });
|
|
||||||
}
|
|
||||||
|
|
||||||
changeTitle() {
|
|
||||||
this.setState({ title: event.target.value });
|
|
||||||
}
|
|
||||||
|
|
||||||
changeDescription() {
|
|
||||||
this.setState({ description: event.target.value });
|
|
||||||
}
|
|
||||||
|
|
||||||
changeColor() {
|
|
||||||
this.setState({ color: event.target.value });
|
|
||||||
}
|
|
||||||
|
|
||||||
submitColor() {
|
|
||||||
const { props, state } = this;
|
|
||||||
|
|
||||||
let color = state.color;
|
|
||||||
if (color.startsWith('#')) {
|
|
||||||
color = state.color.substr(1);
|
|
||||||
}
|
|
||||||
const hexExp = /([0-9A-Fa-f]{6})/;
|
|
||||||
const hexTest = hexExp.exec(color);
|
|
||||||
let currentColor = '000000';
|
|
||||||
if (props.association && 'metadata' in props.association) {
|
|
||||||
currentColor = uxToHex(props.association.metadata.color);
|
|
||||||
}
|
|
||||||
if (hexTest && (hexTest[1] !== currentColor)) {
|
|
||||||
const chatOwner = (deSig(props.match.params.ship) === window.ship);
|
|
||||||
const association =
|
|
||||||
(props.association) && ('metadata' in props.association)
|
|
||||||
? props.association : {};
|
|
||||||
|
|
||||||
if (chatOwner) {
|
|
||||||
this.setState({ awaiting: true, type: 'Editing chat...' }, (() => {
|
|
||||||
props.api.metadata.metadataAdd(
|
|
||||||
'chat',
|
|
||||||
association['app-path'],
|
|
||||||
association['group-path'],
|
|
||||||
association.metadata.title,
|
|
||||||
association.metadata.description,
|
|
||||||
association.metadata['date-created'],
|
|
||||||
color
|
|
||||||
).then(() => {
|
|
||||||
this.setState({ awaiting: false });
|
|
||||||
});
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
deleteChat() {
|
|
||||||
const { props } = this;
|
|
||||||
|
|
||||||
this.setState({
|
this.setState({
|
||||||
isLoading: true,
|
isLoading,
|
||||||
awaiting: true,
|
awaiting,
|
||||||
type: (deSig(props.match.params.ship) === window.ship)
|
type
|
||||||
? 'Deleting chat...'
|
}, closure);
|
||||||
: 'Leaving chat...'
|
|
||||||
}, (() => {
|
|
||||||
props.api.chat.delete(props.station);
|
|
||||||
}));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
groupifyChat() {
|
renderLoading() {
|
||||||
const { props, state } = this;
|
|
||||||
|
|
||||||
this.setState({
|
|
||||||
isLoading: true,
|
|
||||||
awaiting: true,
|
|
||||||
type: 'Converting chat...'
|
|
||||||
}, (() => {
|
|
||||||
props.api.chat.groupify(
|
|
||||||
props.station, state.targetGroup, state.inclusive
|
|
||||||
).then(() => this.setState({ awaiting: false }));
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
|
|
||||||
renderDelete() {
|
|
||||||
const { props } = this;
|
|
||||||
|
|
||||||
const chatOwner = (deSig(props.match.params.ship) === window.ship);
|
|
||||||
|
|
||||||
const deleteButtonClasses = (chatOwner) ? 'b--red2 red2 pointer bg-gray0-d' : 'b--gray3 gray3 bg-gray0-d c-default';
|
|
||||||
const leaveButtonClasses = (!chatOwner) ? 'pointer' : 'c-default';
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<Spinner
|
||||||
<div className={'w-100 fl mt3 ' + ((chatOwner) ? 'o-30' : '')}>
|
awaiting={this.state.awaiting}
|
||||||
<p className="f8 mt3 lh-copy db">Leave Chat</p>
|
classes="absolute right-2 bottom-2 ba pa2 b--gray1-d"
|
||||||
<p className="f9 gray2 db mb4">Remove this chat from your chat list. You will need to request for access again.</p>
|
text={this.state.type}
|
||||||
<a onClick={(!chatOwner) ? this.deleteChat.bind(this) : null}
|
/>
|
||||||
className={'dib f9 black gray4-d bg-gray0-d ba pa2 b--black b--gray1-d ' + leaveButtonClasses}
|
|
||||||
>Leave this chat</a>
|
|
||||||
</div>
|
|
||||||
<div className={'w-100 fl mt3 ' + ((!chatOwner) ? 'o-30' : '')}>
|
|
||||||
<p className="f8 mt3 lh-copy db">Delete Chat</p>
|
|
||||||
<p className="f9 gray2 db mb4">Permanently delete this chat. All current members will no longer see this chat.</p>
|
|
||||||
<a onClick={(chatOwner) ? this.deleteChat.bind(this) : null}
|
|
||||||
className={'dib f9 ba pa2 ' + deleteButtonClasses}
|
|
||||||
>Delete this chat</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
renderGroupify() {
|
renderNormal() {
|
||||||
const { props, state } = this;
|
const { state } = this;
|
||||||
|
const {
|
||||||
|
associations,
|
||||||
|
association,
|
||||||
|
contacts,
|
||||||
|
groups,
|
||||||
|
api,
|
||||||
|
station,
|
||||||
|
match
|
||||||
|
} = this.props;
|
||||||
|
const isOwner = deSig(match.params.ship) === window.ship;
|
||||||
|
|
||||||
const chatOwner = (deSig(props.match.params.ship) === window.ship);
|
return (
|
||||||
|
<Fragment>
|
||||||
const groupPath = props.association['group-path'];
|
<h2 className="f8 pb2">Chat Settings</h2>
|
||||||
const ownedUnmanagedVillage =
|
<GroupifyButton
|
||||||
chatOwner &&
|
isOwner={isOwner}
|
||||||
!props.contacts[groupPath];
|
association={association}
|
||||||
|
associations={associations}
|
||||||
if (!ownedUnmanagedVillage) {
|
contacts={contacts}
|
||||||
return null;
|
groups={groups}
|
||||||
} else {
|
api={api}
|
||||||
let inclusiveToggle = <div />;
|
changeLoading={this.changeLoading} />
|
||||||
if (state.targetGroup) {
|
<DeleteButton
|
||||||
inclusiveToggle = (
|
isOwner={isOwner}
|
||||||
<div className="mt4">
|
changeLoading={this.changeLoading}
|
||||||
<Toggle
|
station={station}
|
||||||
boolean={state.inclusive}
|
api={api} />
|
||||||
change={this.changeInclusive}
|
<MetadataSettings
|
||||||
/>
|
isOwner={isOwner}
|
||||||
<span className="dib f9 white-d inter ml3">
|
changeLoading={this.changeLoading}
|
||||||
Add all members to group
|
api={api}
|
||||||
</span>
|
association={association}
|
||||||
<p className="f9 gray2 pt1" style={{ paddingLeft: 40 }}>
|
station={station} />
|
||||||
Add chat members to the group if they aren't in it yet
|
<Spinner
|
||||||
</p>
|
awaiting={this.state.awaiting}
|
||||||
</div>
|
classes="absolute right-2 bottom-2 ba pa2 b--gray1-d"
|
||||||
);
|
text={this.state.type}
|
||||||
}
|
/>
|
||||||
|
</Fragment>
|
||||||
return (
|
|
||||||
<div>
|
|
||||||
<div className={'w-100 fl mt3'} style={{ maxWidth: '29rem' }}>
|
|
||||||
<p className="f8 mt3 lh-copy db">Convert Chat</p>
|
|
||||||
<p className="f9 gray2 db mb4">
|
|
||||||
Convert this chat into a group with associated chat, or select a
|
|
||||||
group to add this chat to.
|
|
||||||
</p>
|
|
||||||
<InviteSearch
|
|
||||||
groups={props.groups}
|
|
||||||
contacts={props.contacts}
|
|
||||||
associations={props.associations}
|
|
||||||
groupResults={true}
|
|
||||||
shipResults={false}
|
|
||||||
invites={{
|
|
||||||
groups: state.targetGroup ? [state.targetGroup] : [],
|
|
||||||
ships: []
|
|
||||||
}}
|
|
||||||
setInvite={this.changeTargetGroup}
|
|
||||||
/>
|
|
||||||
{inclusiveToggle}
|
|
||||||
<a onClick={this.groupifyChat.bind(this)}
|
|
||||||
className={'dib f9 black gray4-d bg-gray0-d ba pa2 mt4 b--black b--gray1-d pointer'}
|
|
||||||
>
|
|
||||||
Convert to group
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
renderMetadataSettings() {
|
|
||||||
const { props, state } = this;
|
|
||||||
|
|
||||||
const chatOwner = (deSig(props.match.params.ship) === window.ship);
|
|
||||||
|
|
||||||
const association = (props.association) && ('metadata' in props.association)
|
|
||||||
? props.association : {};
|
|
||||||
|
|
||||||
return(
|
|
||||||
<div>
|
|
||||||
<div className={'w-100 pb6 fl mt3 ' + ((chatOwner) ? '' : 'o-30')}>
|
|
||||||
<p className="f8 mt3 lh-copy">Rename</p>
|
|
||||||
<p className="f9 gray2 db mb4">Change the name of this chat</p>
|
|
||||||
<div className="relative w-100 flex"
|
|
||||||
style={{ maxWidth: '29rem' }}
|
|
||||||
>
|
|
||||||
<input
|
|
||||||
className={'f8 ba b--gray3 b--gray2-d bg-gray0-d white-d ' +
|
|
||||||
'focus-b--black focus-b--white-d pa3 db w-100 flex-auto mr3'}
|
|
||||||
value={state.title}
|
|
||||||
disabled={!chatOwner}
|
|
||||||
onChange={this.changeTitle}
|
|
||||||
onBlur={() => {
|
|
||||||
if (chatOwner) {
|
|
||||||
this.setState({ awaiting: true, type: 'Editing chat...' }, (() => {
|
|
||||||
props.api.metadata.metadataAdd(
|
|
||||||
'chat',
|
|
||||||
association['app-path'],
|
|
||||||
association['group-path'],
|
|
||||||
state.title,
|
|
||||||
association.metadata.description,
|
|
||||||
association.metadata['date-created'],
|
|
||||||
uxToHex(association.metadata.color)
|
|
||||||
).then(() => {
|
|
||||||
this.setState({ awaiting: false });
|
|
||||||
});
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<p className="f8 mt3 lh-copy">Change description</p>
|
|
||||||
<p className="f9 gray2 db mb4">Change the description of this chat</p>
|
|
||||||
<div className="relative w-100 flex"
|
|
||||||
style={{ maxWidth: '29rem' }}
|
|
||||||
>
|
|
||||||
<input
|
|
||||||
className={'f8 ba b--gray3 b--gray2-d bg-gray0-d white-d ' +
|
|
||||||
'focus-b--black focus-b--white-d pa3 db w-100 flex-auto mr3'}
|
|
||||||
value={state.description}
|
|
||||||
disabled={!chatOwner}
|
|
||||||
onChange={this.changeDescription}
|
|
||||||
onBlur={() => {
|
|
||||||
if (chatOwner) {
|
|
||||||
this.setState({ awaiting: true, type: 'Editing chat...' }, (() => {
|
|
||||||
props.api.metadata.metadataAdd(
|
|
||||||
'chat',
|
|
||||||
association['app-path'],
|
|
||||||
association['group-path'],
|
|
||||||
association.metadata.title,
|
|
||||||
state.description,
|
|
||||||
association.metadata['date-created'],
|
|
||||||
uxToHex(association.metadata.color)
|
|
||||||
).then(() => {
|
|
||||||
this.setState({ awaiting: false });
|
|
||||||
});
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<p className="f8 mt3 lh-copy">Change color</p>
|
|
||||||
<p className="f9 gray2 db mb4">Give this chat a color when viewing group channels</p>
|
|
||||||
<div className="relative w-100 flex"
|
|
||||||
style={{ maxWidth: '10rem' }}
|
|
||||||
>
|
|
||||||
<div className="absolute"
|
|
||||||
style={{
|
|
||||||
height: 16,
|
|
||||||
width: 16,
|
|
||||||
backgroundColor: state.color,
|
|
||||||
top: 13,
|
|
||||||
left: 11
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
<input
|
|
||||||
className={'pl7 f8 ba b--gray3 b--gray2-d bg-gray0-d white-d ' +
|
|
||||||
'focus-b--black focus-b--white-d pa3 db w-100 flex-auto mr3'}
|
|
||||||
value={state.color}
|
|
||||||
disabled={!chatOwner}
|
|
||||||
onChange={this.changeColor}
|
|
||||||
onBlur={this.submitColor}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
const { props, state } = this;
|
const { state } = this;
|
||||||
const isinPopout = this.props.popout ? 'popout/' : '';
|
const {
|
||||||
|
api,
|
||||||
|
group,
|
||||||
|
association,
|
||||||
|
station,
|
||||||
|
popout,
|
||||||
|
sidebarShown,
|
||||||
|
match,
|
||||||
|
location
|
||||||
|
} = this.props;
|
||||||
|
|
||||||
const permission = Array.from(props.group.members.values());
|
const isInPopout = popout ? "popout/" : "";
|
||||||
|
const title =
|
||||||
if (state.isLoading) {
|
( association &&
|
||||||
let title = props.station.substr(1);
|
('metadata' in association) &&
|
||||||
|
(association.metadata.title !== '')
|
||||||
if ((props.association) && ('metadata' in props.association)) {
|
) ? association.metadata.title : station.substr(1);
|
||||||
title = (props.association.metadata.title !== '')
|
|
||||||
? props.association.metadata.title : props.station.substr(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div className="h-100 w-100 overflow-x-hidden flex flex-column white-d">
|
|
||||||
<div
|
|
||||||
className="w-100 dn-m dn-l dn-xl inter pt4 pb6 pl3 f8"
|
|
||||||
style={{ height: '1rem' }}
|
|
||||||
>
|
|
||||||
<Link to="/~chat/">{'⟵ All Chats'}</Link>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="pl4 pt2 bb b--gray4 b--gray2-d bg-gray0-d flex relative overflow-x-scroll overflow-x-auto-l overflow-x-auto-xl flex-shrink-0"
|
|
||||||
style={{ height: 48 }}
|
|
||||||
>
|
|
||||||
<SidebarSwitcher
|
|
||||||
sidebarShown={this.props.sidebarShown}
|
|
||||||
popout={this.props.popout}
|
|
||||||
/>
|
|
||||||
<Link to={'/~chat/' + isinPopout + 'room' + props.station}
|
|
||||||
className="pt2 white-d"
|
|
||||||
>
|
|
||||||
<h2
|
|
||||||
className={'dib f9 fw4 lh-solid v-top ' +
|
|
||||||
((title === props.station.substr(1)) ? 'mono' : '')}
|
|
||||||
style={{ width: 'max-content' }}
|
|
||||||
>
|
|
||||||
{title}
|
|
||||||
</h2>
|
|
||||||
</Link>
|
|
||||||
<ChatTabBar
|
|
||||||
{...props}
|
|
||||||
station={props.station}
|
|
||||||
numPeers={permission.length}
|
|
||||||
host={props.match.params.ship}
|
|
||||||
api={props.api}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div className="w-100 pl3 mt4 cf">
|
|
||||||
<Spinner awaiting={state.awaiting}
|
|
||||||
classes="absolute right-2 bottom-2 ba pa2 b--gray1-d"
|
|
||||||
text={state.type} />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
let title = props.station.substr(1);
|
|
||||||
|
|
||||||
if ((props.association) && ('metadata' in props.association)) {
|
|
||||||
title = (props.association.metadata.title !== '')
|
|
||||||
? props.association.metadata.title : props.station.substr(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="h-100 w-100 overflow-x-hidden flex flex-column white-d">
|
<div className="h-100 w-100 overflow-x-hidden flex flex-column white-d">
|
||||||
<div
|
<ChatHeader
|
||||||
className="w-100 dn-m dn-l dn-xl inter pt4 pb6 pl3 f8"
|
match={match}
|
||||||
style={{ height: '1rem' }}
|
location={location}
|
||||||
>
|
api={api}
|
||||||
<Link to="/~chat/">{'⟵ All Chats'}</Link>
|
group={group}
|
||||||
</div>
|
association={association}
|
||||||
<div
|
station={station}
|
||||||
className="pl4 pt2 bb b--gray4 b--gray1-d flex relative overflow-x-scroll overflow-x-auto-l overflow-x-auto-xl flex-shrink-0"
|
sidebarShown={sidebarShown}
|
||||||
style={{ height: 48 }}
|
popout={popout} />
|
||||||
>
|
|
||||||
<SidebarSwitcher
|
|
||||||
sidebarShown={this.props.sidebarShown}
|
|
||||||
popout={this.props.popout}
|
|
||||||
api={this.props.api}
|
|
||||||
/>
|
|
||||||
<Link to={'/~chat/' + isinPopout + 'room' + props.station}
|
|
||||||
className="pt2"
|
|
||||||
>
|
|
||||||
<h2
|
|
||||||
className={'dib f9 fw4 lh-solid v-top ' +
|
|
||||||
((title === props.station.substr(1)) ? 'mono' : '')}
|
|
||||||
style={{ width: 'max-content' }}
|
|
||||||
>
|
|
||||||
{title}
|
|
||||||
</h2>
|
|
||||||
</Link>
|
|
||||||
<ChatTabBar
|
|
||||||
{...props}
|
|
||||||
station={props.station}
|
|
||||||
numPeers={permission.length}
|
|
||||||
isOwner={deSig(props.match.params.ship) === window.ship}
|
|
||||||
popout={this.props.popout}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div className="w-100 pl3 mt4 cf">
|
<div className="w-100 pl3 mt4 cf">
|
||||||
<h2 className="f8 pb2">Chat Settings</h2>
|
{(state.isLoading) ? this.renderLoading() : this.renderNormal() }
|
||||||
{this.renderGroupify()}
|
|
||||||
{this.renderDelete()}
|
|
||||||
{this.renderMetadataSettings()}
|
|
||||||
<Spinner awaiting={state.awaiting}
|
|
||||||
classes="absolute right-2 bottom-2 ba pa2 b--gray1-d"
|
|
||||||
text={state.type} />
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
@ -13,8 +13,6 @@ export class Sidebar extends Component {
|
|||||||
render() {
|
render() {
|
||||||
const { props } = this;
|
const { props } = this;
|
||||||
|
|
||||||
const selectedGroups = props.selectedGroups ? props.selectedGroups : [];
|
|
||||||
|
|
||||||
const contactAssoc =
|
const contactAssoc =
|
||||||
(props.associations && 'contacts' in props.associations)
|
(props.associations && 'contacts' in props.associations)
|
||||||
? alphabetiseAssociations(props.associations.contacts) : {};
|
? alphabetiseAssociations(props.associations.contacts) : {};
|
||||||
@ -61,15 +59,6 @@ export class Sidebar extends Component {
|
|||||||
|
|
||||||
const groupedItems = Object.keys(contactAssoc)
|
const groupedItems = Object.keys(contactAssoc)
|
||||||
.filter(each => (groupedChannels[each] || []).length !== 0)
|
.filter(each => (groupedChannels[each] || []).length !== 0)
|
||||||
.filter((each) => {
|
|
||||||
if (selectedGroups.length === 0) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
const selectedPaths = selectedGroups.map((e) => {
|
|
||||||
return e[0];
|
|
||||||
});
|
|
||||||
return selectedPaths.includes(each);
|
|
||||||
})
|
|
||||||
.map((each, i) => {
|
.map((each, i) => {
|
||||||
const channels = groupedChannels[each] || [];
|
const channels = groupedChannels[each] || [];
|
||||||
return(
|
return(
|
||||||
|
@ -48,7 +48,6 @@ export default class GroupsApp extends Component<GroupsAppProps, {}> {
|
|||||||
const invites =
|
const invites =
|
||||||
(Boolean(props.invites) && '/contacts' in props.invites) ?
|
(Boolean(props.invites) && '/contacts' in props.invites) ?
|
||||||
props.invites['/contacts'] : {};
|
props.invites['/contacts'] : {};
|
||||||
const selectedGroups = props.selectedGroups ? props.selectedGroups : [];
|
|
||||||
const s3 = props.s3 ? props.s3 : {};
|
const s3 = props.s3 ? props.s3 : {};
|
||||||
const groups = props.groups || {};
|
const groups = props.groups || {};
|
||||||
const associations = props.associations || {};
|
const associations = props.associations || {};
|
||||||
@ -62,7 +61,6 @@ export default class GroupsApp extends Component<GroupsAppProps, {}> {
|
|||||||
return (
|
return (
|
||||||
<Skeleton
|
<Skeleton
|
||||||
activeDrawer="groups"
|
activeDrawer="groups"
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
history={props.history}
|
history={props.history}
|
||||||
api={api}
|
api={api}
|
||||||
contacts={contacts}
|
contacts={contacts}
|
||||||
@ -86,7 +84,6 @@ export default class GroupsApp extends Component<GroupsAppProps, {}> {
|
|||||||
return (
|
return (
|
||||||
<Skeleton
|
<Skeleton
|
||||||
history={props.history}
|
history={props.history}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
api={api}
|
api={api}
|
||||||
contacts={contacts}
|
contacts={contacts}
|
||||||
groups={groups}
|
groups={groups}
|
||||||
@ -111,7 +108,6 @@ export default class GroupsApp extends Component<GroupsAppProps, {}> {
|
|||||||
return (
|
return (
|
||||||
<Skeleton
|
<Skeleton
|
||||||
history={props.history}
|
history={props.history}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
api={api}
|
api={api}
|
||||||
contacts={contacts}
|
contacts={contacts}
|
||||||
groups={groups}
|
groups={groups}
|
||||||
@ -150,7 +146,6 @@ export default class GroupsApp extends Component<GroupsAppProps, {}> {
|
|||||||
return (
|
return (
|
||||||
<Skeleton
|
<Skeleton
|
||||||
history={props.history}
|
history={props.history}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
api={api}
|
api={api}
|
||||||
contacts={contacts}
|
contacts={contacts}
|
||||||
invites={invites}
|
invites={invites}
|
||||||
@ -198,7 +193,6 @@ export default class GroupsApp extends Component<GroupsAppProps, {}> {
|
|||||||
return (
|
return (
|
||||||
<Skeleton
|
<Skeleton
|
||||||
history={props.history}
|
history={props.history}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
api={api}
|
api={api}
|
||||||
contacts={contacts}
|
contacts={contacts}
|
||||||
groups={groups}
|
groups={groups}
|
||||||
@ -248,7 +242,6 @@ export default class GroupsApp extends Component<GroupsAppProps, {}> {
|
|||||||
<Skeleton
|
<Skeleton
|
||||||
history={props.history}
|
history={props.history}
|
||||||
api={api}
|
api={api}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
contacts={contacts}
|
contacts={contacts}
|
||||||
groups={groups}
|
groups={groups}
|
||||||
invites={invites}
|
invites={invites}
|
||||||
@ -305,7 +298,6 @@ export default class GroupsApp extends Component<GroupsAppProps, {}> {
|
|||||||
<Skeleton
|
<Skeleton
|
||||||
history={props.history}
|
history={props.history}
|
||||||
api={api}
|
api={api}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
contacts={contacts}
|
contacts={contacts}
|
||||||
groups={groups}
|
groups={groups}
|
||||||
invites={invites}
|
invites={invites}
|
||||||
@ -345,7 +337,6 @@ export default class GroupsApp extends Component<GroupsAppProps, {}> {
|
|||||||
<Skeleton
|
<Skeleton
|
||||||
history={props.history}
|
history={props.history}
|
||||||
api={api}
|
api={api}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
contacts={contacts}
|
contacts={contacts}
|
||||||
groups={groups}
|
groups={groups}
|
||||||
invites={invites}
|
invites={invites}
|
||||||
|
@ -103,6 +103,7 @@ export class JoinScreen extends Component {
|
|||||||
spellCheck="false"
|
spellCheck="false"
|
||||||
rows={1}
|
rows={1}
|
||||||
cols={32}
|
cols={32}
|
||||||
|
autoFocus={true}
|
||||||
onKeyPress={(e) => {
|
onKeyPress={(e) => {
|
||||||
if (e.key === 'Enter') {
|
if (e.key === 'Enter') {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
@ -110,7 +111,7 @@ export class JoinScreen extends Component {
|
|||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
style={{
|
style={{
|
||||||
resize: 'none',
|
resize: 'none'
|
||||||
}}
|
}}
|
||||||
onChange={this.groupChange}
|
onChange={this.groupChange}
|
||||||
value={this.state.group}
|
value={this.state.group}
|
||||||
|
@ -72,16 +72,6 @@ export class GroupSidebar extends Component {
|
|||||||
(path in props.groups)
|
(path in props.groups)
|
||||||
);
|
);
|
||||||
})
|
})
|
||||||
.filter((path) => {
|
|
||||||
const selectedGroups = props.selectedGroups ? props.selectedGroups : [];
|
|
||||||
if (selectedGroups.length === 0) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
const selectedPaths = selectedGroups.map(((e) => {
|
|
||||||
return e[0];
|
|
||||||
}));
|
|
||||||
return (selectedPaths.includes(path));
|
|
||||||
})
|
|
||||||
.sort((a, b) => {
|
.sort((a, b) => {
|
||||||
let aName = a.substr(1);
|
let aName = a.substr(1);
|
||||||
let bName = b.substr(1);
|
let bName = b.substr(1);
|
||||||
|
@ -17,7 +17,6 @@ export class Skeleton extends Component {
|
|||||||
invites={props.invites}
|
invites={props.invites}
|
||||||
activeDrawer={props.activeDrawer}
|
activeDrawer={props.activeDrawer}
|
||||||
selected={props.selected}
|
selected={props.selected}
|
||||||
selectedGroups={props.selectedGroups}
|
|
||||||
history={props.history}
|
history={props.history}
|
||||||
api={props.api}
|
api={props.api}
|
||||||
associations={props.associations}
|
associations={props.associations}
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import classnames from 'classnames';
|
import classnames from 'classnames';
|
||||||
import { Link } from 'react-router-dom';
|
import { Link } from 'react-router-dom';
|
||||||
|
import defaultApps from '../../../../lib/default-apps';
|
||||||
|
|
||||||
import Tile from './tile';
|
import Tile from './tile';
|
||||||
|
|
||||||
@ -29,7 +30,9 @@ export default class BasicTile extends React.PureComponent {
|
|||||||
</span>
|
</span>
|
||||||
);
|
);
|
||||||
|
|
||||||
const routeList = ['/~chat', '/~publish', '/~link', '/~groups', '/~dojo'];
|
const routeList = defaultApps.map((e) => {
|
||||||
|
return `/~${e}`;
|
||||||
|
});
|
||||||
|
|
||||||
const tile = ( routeList.indexOf(props.linkedUrl) !== -1 ) ? (
|
const tile = ( routeList.indexOf(props.linkedUrl) !== -1 ) ? (
|
||||||
<Link className="w-100 h-100 db pa2 no-underline" to={props.linkedUrl}>
|
<Link className="w-100 h-100 db pa2 no-underline" to={props.linkedUrl}>
|
||||||
|
@ -41,7 +41,7 @@ export class LinksApp extends Component {
|
|||||||
render() {
|
render() {
|
||||||
const { props } = this;
|
const { props } = this;
|
||||||
|
|
||||||
const contacts = props.contacts ? props.contacts : {};
|
const contacts = props.contacts ? props.contacts : {};
|
||||||
|
|
||||||
const groups = props.groups ? props.groups : {};
|
const groups = props.groups ? props.groups : {};
|
||||||
|
|
||||||
@ -51,18 +51,9 @@ export class LinksApp extends Component {
|
|||||||
|
|
||||||
const seen = props.linksSeen ? props.linksSeen : {};
|
const seen = props.linksSeen ? props.linksSeen : {};
|
||||||
|
|
||||||
const selectedGroups = props.selectedGroups ? props.selectedGroups : [];
|
|
||||||
|
|
||||||
const selGroupPaths = selectedGroups.map(g => g[0]);
|
|
||||||
const totalUnseen = _.reduce(
|
const totalUnseen = _.reduce(
|
||||||
links,
|
links,
|
||||||
(acc, collection, path) => {
|
(acc, collection) => acc + collection.unseenCount,
|
||||||
if(selGroupPaths.length > 0
|
|
||||||
&& !selGroupPaths.includes(associations.link?.[path]?.['group-path'])) {
|
|
||||||
return acc;
|
|
||||||
}
|
|
||||||
return acc + collection.unseenCount;
|
|
||||||
},
|
|
||||||
0
|
0
|
||||||
);
|
);
|
||||||
|
|
||||||
@ -91,7 +82,6 @@ export class LinksApp extends Component {
|
|||||||
groups={groups}
|
groups={groups}
|
||||||
rightPanelHide={true}
|
rightPanelHide={true}
|
||||||
sidebarShown={sidebarShown}
|
sidebarShown={sidebarShown}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
links={links}
|
links={links}
|
||||||
listening={listening}
|
listening={listening}
|
||||||
api={api}
|
api={api}
|
||||||
@ -109,7 +99,6 @@ export class LinksApp extends Component {
|
|||||||
invites={invites}
|
invites={invites}
|
||||||
groups={groups}
|
groups={groups}
|
||||||
sidebarShown={sidebarShown}
|
sidebarShown={sidebarShown}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
links={links}
|
links={links}
|
||||||
listening={listening}
|
listening={listening}
|
||||||
api={api}
|
api={api}
|
||||||
@ -157,7 +146,6 @@ export class LinksApp extends Component {
|
|||||||
groups={groups}
|
groups={groups}
|
||||||
selected={resourcePath}
|
selected={resourcePath}
|
||||||
sidebarShown={sidebarShown}
|
sidebarShown={sidebarShown}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
links={links}
|
links={links}
|
||||||
listening={listening}
|
listening={listening}
|
||||||
api={api}
|
api={api}
|
||||||
@ -198,7 +186,6 @@ export class LinksApp extends Component {
|
|||||||
groups={groups}
|
groups={groups}
|
||||||
selected={resourcePath}
|
selected={resourcePath}
|
||||||
sidebarShown={sidebarShown}
|
sidebarShown={sidebarShown}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
popout={popout}
|
popout={popout}
|
||||||
links={links}
|
links={links}
|
||||||
listening={listening}
|
listening={listening}
|
||||||
@ -253,7 +240,6 @@ export class LinksApp extends Component {
|
|||||||
groups={groups}
|
groups={groups}
|
||||||
selected={resourcePath}
|
selected={resourcePath}
|
||||||
sidebarShown={sidebarShown}
|
sidebarShown={sidebarShown}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
sidebarHideMobile={true}
|
sidebarHideMobile={true}
|
||||||
popout={popout}
|
popout={popout}
|
||||||
links={links}
|
links={links}
|
||||||
@ -311,7 +297,6 @@ export class LinksApp extends Component {
|
|||||||
groups={groups}
|
groups={groups}
|
||||||
selected={resourcePath}
|
selected={resourcePath}
|
||||||
sidebarShown={sidebarShown}
|
sidebarShown={sidebarShown}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
sidebarHideMobile={true}
|
sidebarHideMobile={true}
|
||||||
popout={popout}
|
popout={popout}
|
||||||
links={links}
|
links={links}
|
||||||
|
@ -51,24 +51,14 @@ export class ChannelsSidebar extends Component {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
const selectedGroups = props.selectedGroups ? props.selectedGroups : [];
|
|
||||||
let i = -1;
|
let i = -1;
|
||||||
const groupedItems = Object.keys(associations)
|
const groupedItems = Object.keys(associations)
|
||||||
.filter((each) => {
|
|
||||||
if (selectedGroups.length === 0) {
|
|
||||||
return true;
|
|
||||||
};
|
|
||||||
const selectedPaths = selectedGroups.map((e) => {
|
|
||||||
return e[0];
|
|
||||||
});
|
|
||||||
return selectedPaths.includes(each);
|
|
||||||
})
|
|
||||||
.map((each) => {
|
.map((each) => {
|
||||||
const channels = groupedChannels[each];
|
const channels = groupedChannels[each];
|
||||||
if (!channels || channels.length === 0)
|
if (!channels || channels.length === 0)
|
||||||
return;
|
return;
|
||||||
i++;
|
i++;
|
||||||
if ((selectedGroups.length === 0) && groupedChannels['/~/'] && groupedChannels['/~/'].length !== 0) {
|
if (groupedChannels['/~/'] && groupedChannels['/~/'].length !== 0) {
|
||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -84,7 +74,7 @@ export class ChannelsSidebar extends Component {
|
|||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
if ((selectedGroups.length === 0) && groupedChannels['/~/'] && groupedChannels['/~/'].length !== 0) {
|
if (groupedChannels['/~/'] && groupedChannels['/~/'].length !== 0) {
|
||||||
groupedItems.unshift(
|
groupedItems.unshift(
|
||||||
<GroupItem
|
<GroupItem
|
||||||
key={'/~/'}
|
key={'/~/'}
|
||||||
|
@ -31,7 +31,6 @@ export class Skeleton extends Component {
|
|||||||
invites={linkInvites}
|
invites={linkInvites}
|
||||||
groups={props.groups}
|
groups={props.groups}
|
||||||
selected={props.selected}
|
selected={props.selected}
|
||||||
selectedGroups={props.selectedGroups}
|
|
||||||
sidebarShown={props.sidebarShown}
|
sidebarShown={props.sidebarShown}
|
||||||
links={props.links}
|
links={props.links}
|
||||||
listening={props.listening}
|
listening={props.listening}
|
||||||
|
@ -42,7 +42,6 @@ export default class PublishApp extends React.Component {
|
|||||||
|
|
||||||
const contacts = props.contacts ? props.contacts : {};
|
const contacts = props.contacts ? props.contacts : {};
|
||||||
const associations = props.associations ? props.associations : { contacts: {} };
|
const associations = props.associations ? props.associations : { contacts: {} };
|
||||||
const selectedGroups = props.selectedGroups ? props.selectedGroups : [];
|
|
||||||
|
|
||||||
const notebooks = props.notebooks ? props.notebooks : {};
|
const notebooks = props.notebooks ? props.notebooks : {};
|
||||||
|
|
||||||
@ -50,12 +49,6 @@ export default class PublishApp extends React.Component {
|
|||||||
.values()
|
.values()
|
||||||
.map(_.values)
|
.map(_.values)
|
||||||
.flatten() // flatten into array of notebooks
|
.flatten() // flatten into array of notebooks
|
||||||
.filter((each) => {
|
|
||||||
return ((selectedGroups.map((e) => {
|
|
||||||
return e[0];
|
|
||||||
}).includes(each?.['writers-group-path'])) ||
|
|
||||||
(selectedGroups.length === 0));
|
|
||||||
})
|
|
||||||
.map('num-unread')
|
.map('num-unread')
|
||||||
.reduce((acc, count) => acc + count, 0)
|
.reduce((acc, count) => acc + count, 0)
|
||||||
.value();
|
.value();
|
||||||
@ -80,7 +73,6 @@ export default class PublishApp extends React.Component {
|
|||||||
invites={invites}
|
invites={invites}
|
||||||
notebooks={notebooks}
|
notebooks={notebooks}
|
||||||
associations={associations}
|
associations={associations}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
contacts={contacts}
|
contacts={contacts}
|
||||||
api={api}
|
api={api}
|
||||||
>
|
>
|
||||||
@ -111,7 +103,6 @@ export default class PublishApp extends React.Component {
|
|||||||
invites={invites}
|
invites={invites}
|
||||||
notebooks={notebooks}
|
notebooks={notebooks}
|
||||||
associations={associations}
|
associations={associations}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
contacts={contacts}
|
contacts={contacts}
|
||||||
api={api}
|
api={api}
|
||||||
>
|
>
|
||||||
@ -142,7 +133,6 @@ export default class PublishApp extends React.Component {
|
|||||||
invites={invites}
|
invites={invites}
|
||||||
notebooks={notebooks}
|
notebooks={notebooks}
|
||||||
associations={associations}
|
associations={associations}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
contacts={contacts}
|
contacts={contacts}
|
||||||
api={api}
|
api={api}
|
||||||
>
|
>
|
||||||
@ -188,7 +178,6 @@ export default class PublishApp extends React.Component {
|
|||||||
invites={invites}
|
invites={invites}
|
||||||
notebooks={notebooks}
|
notebooks={notebooks}
|
||||||
associations={associations}
|
associations={associations}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
contacts={contacts}
|
contacts={contacts}
|
||||||
path={path}
|
path={path}
|
||||||
api={api}
|
api={api}
|
||||||
@ -215,7 +204,6 @@ export default class PublishApp extends React.Component {
|
|||||||
notebooks={notebooks}
|
notebooks={notebooks}
|
||||||
associations={associations}
|
associations={associations}
|
||||||
contacts={contacts}
|
contacts={contacts}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
path={path}
|
path={path}
|
||||||
api={api}
|
api={api}
|
||||||
>
|
>
|
||||||
@ -265,7 +253,6 @@ export default class PublishApp extends React.Component {
|
|||||||
sidebarShown={sidebarShown}
|
sidebarShown={sidebarShown}
|
||||||
invites={invites}
|
invites={invites}
|
||||||
notebooks={notebooks}
|
notebooks={notebooks}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
associations={associations}
|
associations={associations}
|
||||||
contacts={contacts}
|
contacts={contacts}
|
||||||
path={path}
|
path={path}
|
||||||
@ -293,7 +280,6 @@ export default class PublishApp extends React.Component {
|
|||||||
invites={invites}
|
invites={invites}
|
||||||
notebooks={notebooks}
|
notebooks={notebooks}
|
||||||
associations={associations}
|
associations={associations}
|
||||||
selectedGroups={selectedGroups}
|
|
||||||
contacts={contacts}
|
contacts={contacts}
|
||||||
path={path}
|
path={path}
|
||||||
api={api}
|
api={api}
|
||||||
|
@ -12,7 +12,8 @@ export class Note extends Component {
|
|||||||
constructor(props) {
|
constructor(props) {
|
||||||
super(props);
|
super(props);
|
||||||
this.state = {
|
this.state = {
|
||||||
deleting: false
|
deleting: false,
|
||||||
|
sentRead: false
|
||||||
};
|
};
|
||||||
moment.updateLocale('en', {
|
moment.updateLocale('en', {
|
||||||
relativeTime: {
|
relativeTime: {
|
||||||
@ -46,16 +47,19 @@ export class Note extends Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
componentDidUpdate(prevProps) {
|
componentDidUpdate(prevProps) {
|
||||||
const { props } = this;
|
const { props, state } = this;
|
||||||
if ((prevProps && prevProps.api !== props.api) || props.api) {
|
if ((prevProps && prevProps.api !== props.api) || props.api) {
|
||||||
if (!(props.notebooks[props.ship]?.[props.book]?.notes?.[props.note]?.file)) {
|
if (!(props.notebooks[props.ship]?.[props.book]?.notes?.[props.note]?.file)) {
|
||||||
props.api.publish.fetchNote(props.ship, props.book, props.note);
|
props.api.publish.fetchNote(props.ship, props.book, props.note);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (prevProps) {
|
if (prevProps && prevProps.note !== props.note) {
|
||||||
if ((prevProps.book !== props.book) ||
|
this.setState({ sentRead: false });
|
||||||
(prevProps.note !== props.note) ||
|
}
|
||||||
(prevProps.ship !== props.ship)) {
|
|
||||||
|
if (!state.sentRead &&
|
||||||
|
props.notebooks?.[props.ship]?.[props.book]?.notes?.[props.note] &&
|
||||||
|
!props.notebooks[props.ship][props.book].notes[props.note].read) {
|
||||||
const readAction = {
|
const readAction = {
|
||||||
read: {
|
read: {
|
||||||
who: props.ship.slice(1),
|
who: props.ship.slice(1),
|
||||||
@ -63,9 +67,10 @@ export class Note extends Component {
|
|||||||
note: props.note
|
note: props.note
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
props.api.publish.publishAction(readAction);
|
this.setState({ sentRead: true }, () => {
|
||||||
|
props.api.publish.publishAction(readAction);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -84,7 +84,7 @@ export class NotebookPosts extends Component {
|
|||||||
' gray2 mr3'}
|
' gray2 mr3'}
|
||||||
title={note.author}
|
title={note.author}
|
||||||
>{name}</div>
|
>{name}</div>
|
||||||
<div className="gray2 mr3">{date}</div>
|
<div className={((note.read) ? "gray2 " : "green2 ") + "mr3"}>{date}</div>
|
||||||
<div className="gray2">{comment}</div>
|
<div className="gray2">{comment}</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -64,23 +64,12 @@ export class Sidebar extends Component {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
const selectedGroups = props.selectedGroups ? props.selectedGroups: [];
|
|
||||||
const groupedItems = Object.keys(associations)
|
const groupedItems = Object.keys(associations)
|
||||||
.filter((each) => {
|
|
||||||
if (selectedGroups.length === 0) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
const selectedPaths = selectedGroups.map((e) => {
|
|
||||||
return e[0];
|
|
||||||
});
|
|
||||||
return (selectedPaths.includes(each));
|
|
||||||
})
|
|
||||||
.map((each, i) => {
|
.map((each, i) => {
|
||||||
const books = groupedNotebooks[each] || [];
|
const books = groupedNotebooks[each] || [];
|
||||||
if (books.length === 0)
|
if (books.length === 0)
|
||||||
return;
|
return;
|
||||||
if ((selectedGroups.length === 0) &&
|
if (groupedNotebooks['/~/'] &&
|
||||||
groupedNotebooks['/~/'] &&
|
|
||||||
groupedNotebooks['/~/'].length !== 0) {
|
groupedNotebooks['/~/'].length !== 0) {
|
||||||
i = i + 1;
|
i = i + 1;
|
||||||
}
|
}
|
||||||
@ -95,8 +84,7 @@ export class Sidebar extends Component {
|
|||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
if ((selectedGroups.length === 0) &&
|
if (groupedNotebooks['/~/'] &&
|
||||||
groupedNotebooks['/~/'] &&
|
|
||||||
groupedNotebooks['/~/'].length !== 0) {
|
groupedNotebooks['/~/'].length !== 0) {
|
||||||
groupedItems.unshift(
|
groupedItems.unshift(
|
||||||
<GroupItem
|
<GroupItem
|
||||||
|
@ -30,7 +30,6 @@ export class Skeleton extends Component {
|
|||||||
path={props.path}
|
path={props.path}
|
||||||
invites={props.invites}
|
invites={props.invites}
|
||||||
associations={props.associations}
|
associations={props.associations}
|
||||||
selectedGroups={props.selectedGroups}
|
|
||||||
api={this.props.api}
|
api={this.props.api}
|
||||||
/>
|
/>
|
||||||
<div className={'h-100 w-100 relative white-d flex-auto ' + rightPanelHide} style={{
|
<div className={'h-100 w-100 relative white-d flex-auto ' + rightPanelHide} style={{
|
||||||
|
@ -1,249 +0,0 @@
|
|||||||
import React, { Component } from 'react';
|
|
||||||
import { Link } from 'react-router-dom';
|
|
||||||
|
|
||||||
|
|
||||||
export default class GroupFilter extends Component {
|
|
||||||
constructor(props) {
|
|
||||||
super(props);
|
|
||||||
this.state = {
|
|
||||||
open: false,
|
|
||||||
selected: [],
|
|
||||||
groups: [],
|
|
||||||
searchTerm: '',
|
|
||||||
results: []
|
|
||||||
};
|
|
||||||
this.toggleOpen = this.toggleOpen.bind(this);
|
|
||||||
this.handleClickOutside = this.handleClickOutside.bind(this);
|
|
||||||
this.groupIndex = this.groupIndex.bind(this);
|
|
||||||
this.search = this.search.bind(this);
|
|
||||||
this.addGroup = this.addGroup.bind(this);
|
|
||||||
this.deleteGroup = this.deleteGroup.bind(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
componentDidMount() {
|
|
||||||
document.addEventListener('mousedown', this.handleClickOutside);
|
|
||||||
this.groupIndex();
|
|
||||||
const selected = localStorage.getItem('urbit-selectedGroups');
|
|
||||||
if (selected) {
|
|
||||||
this.setState({ selected: JSON.parse(selected) }, (() => {
|
|
||||||
this.props.api.local.setSelected(this.state.selected);
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
componentWillUnmount() {
|
|
||||||
document.removeEventListener('mousedown', this.handleClickOutside);
|
|
||||||
}
|
|
||||||
|
|
||||||
componentDidUpdate(prevProps) {
|
|
||||||
if (prevProps !== this.props) {
|
|
||||||
this.groupIndex();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
handleClickOutside(evt) {
|
|
||||||
if ((this.dropdown && !this.dropdown.contains(evt.target))
|
|
||||||
&& (this.toggleButton && !this.toggleButton.contains(evt.target))) {
|
|
||||||
this.setState({ open: false });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
toggleOpen() {
|
|
||||||
this.setState({ open: !this.state.open });
|
|
||||||
}
|
|
||||||
|
|
||||||
groupIndex() {
|
|
||||||
const { props } = this;
|
|
||||||
let index = [];
|
|
||||||
const associations =
|
|
||||||
(props.associations && 'contacts' in props.associations) ?
|
|
||||||
props.associations.contacts : {};
|
|
||||||
index = Object.keys(associations).map((each) => {
|
|
||||||
const eachGroup = [];
|
|
||||||
eachGroup.push(each);
|
|
||||||
let name = each;
|
|
||||||
if (associations[each].metadata) {
|
|
||||||
name = (associations[each].metadata.title !== '')
|
|
||||||
? associations[each].metadata.title : name;
|
|
||||||
}
|
|
||||||
eachGroup.push(name);
|
|
||||||
return eachGroup;
|
|
||||||
});
|
|
||||||
this.setState({ groups: index });
|
|
||||||
}
|
|
||||||
|
|
||||||
search(evt) {
|
|
||||||
this.setState({ searchTerm: evt.target.value });
|
|
||||||
const term = evt.target.value.toLowerCase();
|
|
||||||
|
|
||||||
if (term.length < 3) {
|
|
||||||
return this.setState({ results: [] });
|
|
||||||
}
|
|
||||||
|
|
||||||
let groupMatches = [];
|
|
||||||
groupMatches = this.state.groups.filter((e) => {
|
|
||||||
return (e[0].includes(term) || e[1].includes(term));
|
|
||||||
});
|
|
||||||
this.setState({ results: groupMatches });
|
|
||||||
}
|
|
||||||
|
|
||||||
addGroup(group) {
|
|
||||||
const selected = this.state.selected;
|
|
||||||
if (!(group in selected)) {
|
|
||||||
selected.push(group);
|
|
||||||
}
|
|
||||||
this.setState({
|
|
||||||
searchTerm: '',
|
|
||||||
selected: selected,
|
|
||||||
results: []
|
|
||||||
}, (() => {
|
|
||||||
this.props.api.local.setSelected(this.state.selected);
|
|
||||||
localStorage.setItem('urbit-selectedGroups', JSON.stringify(this.state.selected));
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
|
|
||||||
deleteGroup(group) {
|
|
||||||
let selected = this.state.selected;
|
|
||||||
selected = selected.filter((e) => {
|
|
||||||
return e !== group;
|
|
||||||
});
|
|
||||||
this.setState({ selected: selected }, (() => {
|
|
||||||
this.props.api.local.setSelected(this.state.selected);
|
|
||||||
localStorage.setItem('urbit-selectedGroups', JSON.stringify(this.state.selected));
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
|
|
||||||
render() {
|
|
||||||
const { props, state } = this;
|
|
||||||
|
|
||||||
let currentGroup = 'All Groups';
|
|
||||||
|
|
||||||
if (state.selected.length > 0) {
|
|
||||||
const titles = state.selected.map((each) => {
|
|
||||||
return each[1];
|
|
||||||
});
|
|
||||||
currentGroup = titles.join(' + ');
|
|
||||||
}
|
|
||||||
|
|
||||||
const buttonOpened = (state.open)
|
|
||||||
? 'bg-gray5 bg-gray1-d white-d' : 'hover-bg-gray5 hover-bg-gray1-d white-d';
|
|
||||||
|
|
||||||
const dropdownClass = (state.open)
|
|
||||||
? 'absolute db z-2 bg-white bg-gray0-d white-d ba b--gray3 b--gray1-d'
|
|
||||||
: 'dn';
|
|
||||||
|
|
||||||
const inviteCount = (props.invites && Object.keys(props.invites).length > 0)
|
|
||||||
? <template className="dib fr">
|
|
||||||
<p className="dib bg-green2 bg-gray2-d white fw6 ph1 br1 v-mid" style={{ marginBottom: 2 }}>
|
|
||||||
{Object.keys(props.invites).length}
|
|
||||||
</p>
|
|
||||||
<span className="dib v-mid ml1">
|
|
||||||
<img
|
|
||||||
className="v-mid"
|
|
||||||
src="/~landscape/img/Chevron.png"
|
|
||||||
style={{ height: 16, width: 16, paddingBottom: 1 }}
|
|
||||||
/>
|
|
||||||
</span>
|
|
||||||
</template>
|
|
||||||
: <template className="dib fr">
|
|
||||||
<span className="dib v-top ml1">
|
|
||||||
<img className="v-mid"
|
|
||||||
src="/~landscape/img/Chevron.png"
|
|
||||||
style={{ height: 16, width: 16, paddingBottom: 1 }}
|
|
||||||
/>
|
|
||||||
</span>
|
|
||||||
</template>;
|
|
||||||
|
|
||||||
let selectedGroups = <div />;
|
|
||||||
let searchResults = <div />;
|
|
||||||
|
|
||||||
if (state.results.length > 0) {
|
|
||||||
const groupResults = state.results.map(((group) => {
|
|
||||||
return(
|
|
||||||
<li
|
|
||||||
key={group[0]}
|
|
||||||
className="tl list white-d f9 pv2 ph3 pointer hover-bg-gray4 hover-bg-gray1-d inter"
|
|
||||||
onClick={() => this.addGroup(group)}
|
|
||||||
>
|
|
||||||
<span className="mix-blend-diff white">{(group[1]) ? group[1] : group[0]}</span>
|
|
||||||
</li>
|
|
||||||
);
|
|
||||||
}));
|
|
||||||
searchResults = (
|
|
||||||
<div className={'tl absolute bg-white bg-gray0-d white-d pv3 z-1 w-100 ba b--gray4 b--white-d overflow-y-scroll'} style={{ maxWidth: '15.67rem', maxHeight: '8rem' }}>
|
|
||||||
<p className="f9 tl gray2 ph3 pb2">Groups</p>
|
|
||||||
{groupResults}
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (state.selected.length > 0) {
|
|
||||||
const allSelected = this.state.selected.map((each) => {
|
|
||||||
const name = each[1];
|
|
||||||
return(
|
|
||||||
<span
|
|
||||||
key={each[0]}
|
|
||||||
className={'f9 inter black pa2 bg-gray5 bg-gray1-d ' +
|
|
||||||
'ba b--gray4 b--gray2-d white-d dib mr2 mt2 c-default'}
|
|
||||||
>
|
|
||||||
{name}
|
|
||||||
<span
|
|
||||||
className="white-d ml3 mono pointer"
|
|
||||||
onClick={e => this.deleteGroup(each)}
|
|
||||||
>
|
|
||||||
x
|
|
||||||
</span>
|
|
||||||
</span>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
selectedGroups = (
|
|
||||||
<div className={
|
|
||||||
'f9 gray2 bb bl br b--gray3 b--gray2-d bg-gray0-d ' +
|
|
||||||
'white-d pa3 db w-100 inter bg-gray5 lh-solid tl'
|
|
||||||
}
|
|
||||||
style={{ width: 251 }}
|
|
||||||
>
|
|
||||||
{allSelected}
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div className="ml1 dib">
|
|
||||||
<div className={buttonOpened}
|
|
||||||
onClick={() => this.toggleOpen()}
|
|
||||||
ref={el => this.toggleButton = el}
|
|
||||||
>
|
|
||||||
<p className="inter dib f9 pointer pv1 ph2 mw5 truncate v-mid">{currentGroup}</p>
|
|
||||||
</div>
|
|
||||||
<div className={dropdownClass}
|
|
||||||
style={{ maxHeight: '24rem', width: 285 }}
|
|
||||||
ref={(el) => {
|
|
||||||
this.dropdown = el;
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<p className="tc bb b--gray3 b--gray1-d gray3 pv4 f9">Group Select and Filter</p>
|
|
||||||
<Link to="/~groups"
|
|
||||||
className="ma4 bg-gray5 bg-gray1-d f9 tl pa1 br1 db no-underline"
|
|
||||||
style={{ paddingLeft: '6.5px', paddingRight: '6.5px' }}
|
|
||||||
onClick={() => this.setState({ open: false })}
|
|
||||||
>
|
|
||||||
Manage all Groups
|
|
||||||
{inviteCount}
|
|
||||||
</Link>
|
|
||||||
<p className="pt4 gray3 f9 tl mh4">Filter Groups</p>
|
|
||||||
<div className="relative w-100 ph4 pt2 pb4">
|
|
||||||
<input className="ba b--gray3 white-d bg-gray0-d inter w-100 f9 pa2"
|
|
||||||
style={{ boxSizing: 'border-box' }}
|
|
||||||
placeholder="Group name..."
|
|
||||||
onChange={this.search}
|
|
||||||
value={state.searchTerm}
|
|
||||||
/>
|
|
||||||
{searchResults}
|
|
||||||
{selectedGroups}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
261
pkg/interface/src/components/Omnibox.js
Normal file
261
pkg/interface/src/components/Omnibox.js
Normal file
@ -0,0 +1,261 @@
|
|||||||
|
import React, { Component } from 'react';
|
||||||
|
import { withRouter } from 'react-router-dom';
|
||||||
|
import { Box, Row, Rule, Text } from '@tlon/indigo-react';
|
||||||
|
import index from '../lib/omnibox';
|
||||||
|
import Mousetrap from 'mousetrap';
|
||||||
|
import OmniboxInput from './OmniboxInput';
|
||||||
|
import OmniboxResult from './OmniboxResult';
|
||||||
|
|
||||||
|
import { cite } from '../lib/util';
|
||||||
|
|
||||||
|
export class Omnibox extends Component {
|
||||||
|
constructor(props) {
|
||||||
|
super(props);
|
||||||
|
this.state = {
|
||||||
|
index: new Map([]),
|
||||||
|
query: '',
|
||||||
|
results: this.initialResults(),
|
||||||
|
selected: ''
|
||||||
|
};
|
||||||
|
this.handleClickOutside = this.handleClickOutside.bind(this);
|
||||||
|
this.search = this.search.bind(this);
|
||||||
|
this.navigate = this.navigate.bind(this);
|
||||||
|
this.control - this.control.bind(this);
|
||||||
|
this.setPreviousSelected = this.setPreviousSelected.bind(this);
|
||||||
|
this.setNextSelected = this.setNextSelected.bind(this);
|
||||||
|
this.renderResults = this.renderResults.bind(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidUpdate(prevProps) {
|
||||||
|
if (prevProps !== this.props) {
|
||||||
|
this.setState({ index: index(this.props.associations, this.props.apps.tiles) });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (prevProps && this.props.show && prevProps.show !== this.props.show) {
|
||||||
|
Mousetrap.bind('escape', () => this.props.api.local.setOmnibox());
|
||||||
|
document.addEventListener('mousedown', this.handleClickOutside);
|
||||||
|
const touchstart = new Event('touchstart');
|
||||||
|
this.omniInput.input.dispatchEvent(touchstart);
|
||||||
|
this.omniInput.input.focus();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
componentWillUpdate(prevProps) {
|
||||||
|
if (this.props.show && prevProps.show !== this.props.show) {
|
||||||
|
Mousetrap.unbind('escape');
|
||||||
|
document.removeEventListener('mousedown', this.handleClickOutside);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
getSearchedCategories() {
|
||||||
|
return ['apps', 'commands', 'groups', 'subscriptions'];
|
||||||
|
}
|
||||||
|
|
||||||
|
control(evt) {
|
||||||
|
if (evt.key === 'Escape') {
|
||||||
|
if (this.state.query.length > 0) {
|
||||||
|
this.setState({ query: '', results: this.initialResults() });
|
||||||
|
} else if (this.props.show) {
|
||||||
|
this.props.api.local.setOmnibox();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if (
|
||||||
|
evt.key === 'ArrowUp' ||
|
||||||
|
(evt.shiftKey && evt.key === 'Tab')) {
|
||||||
|
evt.preventDefault();
|
||||||
|
return this.setPreviousSelected();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (evt.key === 'ArrowDown' || evt.key === 'Tab') {
|
||||||
|
evt.preventDefault();
|
||||||
|
this.setNextSelected();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (evt.key === 'Enter') {
|
||||||
|
evt.preventDefault();
|
||||||
|
if (this.state.selected !== '') {
|
||||||
|
this.navigate(this.state.selected);
|
||||||
|
} else {
|
||||||
|
this.navigate(Array.from(this.state.results.values()).flat()[0].link);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
handleClickOutside(evt) {
|
||||||
|
if (this.props.show && !this.omniBox.contains(evt.target)) {
|
||||||
|
this.setState({ results: this.initialResults(), query: '' }, () => {
|
||||||
|
this.props.api.local.setOmnibox();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
initialResults() {
|
||||||
|
return new Map(this.getSearchedCategories().map(category => [category, []]));
|
||||||
|
}
|
||||||
|
|
||||||
|
navigate(link) {
|
||||||
|
const { props } = this;
|
||||||
|
this.setState({ results: this.initialResults(), query: '' }, () => {
|
||||||
|
props.api.local.setOmnibox();
|
||||||
|
props.history.push(link);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
search(event) {
|
||||||
|
const { state } = this;
|
||||||
|
let query = event.target.value;
|
||||||
|
const results = this.initialResults();
|
||||||
|
let selected = state.selected;
|
||||||
|
|
||||||
|
this.setState({ query });
|
||||||
|
|
||||||
|
// wipe results if backspacing
|
||||||
|
if (query.length === 0) {
|
||||||
|
this.setState({ results: results, selected: '' });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// don't search for single characters
|
||||||
|
if (query.length === 1) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
query = query.toLowerCase();
|
||||||
|
|
||||||
|
this.getSearchedCategories().map((category) => {
|
||||||
|
const categoryIndex = state.index.get(category);
|
||||||
|
results.set(category,
|
||||||
|
categoryIndex.filter((result) => {
|
||||||
|
return (
|
||||||
|
result.title.toLowerCase().includes(query) ||
|
||||||
|
result.link.toLowerCase().includes(query) ||
|
||||||
|
result.app.toLowerCase().includes(query) ||
|
||||||
|
(result.host !== null ? result.host.includes(query) : false)
|
||||||
|
);
|
||||||
|
})
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
const flattenedResultLinks = Array.from(results.values()).flat().map(result => result.link);
|
||||||
|
if (!flattenedResultLinks.includes(selected)) {
|
||||||
|
selected = flattenedResultLinks[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
this.setState({ results, selected });
|
||||||
|
}
|
||||||
|
|
||||||
|
setPreviousSelected() {
|
||||||
|
const current = this.state.selected;
|
||||||
|
const flattenedResults = Array.from(this.state.results.values()).flat();
|
||||||
|
const totalLength = flattenedResults.length;
|
||||||
|
if (current !== '') {
|
||||||
|
const currentIndex = flattenedResults.indexOf(
|
||||||
|
...flattenedResults.filter((e) => {
|
||||||
|
return e.link === current;
|
||||||
|
})
|
||||||
|
);
|
||||||
|
if (currentIndex > 0) {
|
||||||
|
const nextLink = flattenedResults[currentIndex - 1].link;
|
||||||
|
this.setState({ selected: nextLink });
|
||||||
|
} else {
|
||||||
|
const nextLink = flattenedResults[totalLength - 1].link;
|
||||||
|
this.setState({ selected: nextLink });
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
const nextLink = flattenedResults[totalLength - 1].link;
|
||||||
|
this.setState({ selected: nextLink });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setNextSelected() {
|
||||||
|
const current = this.state.selected;
|
||||||
|
const flattenedResults = Array.from(this.state.results.values()).flat();
|
||||||
|
if (current !== '') {
|
||||||
|
const currentIndex = flattenedResults.indexOf(
|
||||||
|
...flattenedResults.filter((e) => {
|
||||||
|
return e.link === current;
|
||||||
|
})
|
||||||
|
);
|
||||||
|
if (currentIndex < flattenedResults.length - 1) {
|
||||||
|
const nextLink = flattenedResults[currentIndex + 1].link;
|
||||||
|
this.setState({ selected: nextLink });
|
||||||
|
} else {
|
||||||
|
const nextLink = flattenedResults[0].link;
|
||||||
|
this.setState({ selected: nextLink });
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
const nextLink = flattenedResults[0].link;
|
||||||
|
this.setState({ selected: nextLink });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
renderResults() {
|
||||||
|
const { props, state } = this;
|
||||||
|
return <Box maxHeight="400px" overflowY="scroll" overflowX="hidden">
|
||||||
|
{this.getSearchedCategories()
|
||||||
|
.map(category => Object({ category, categoryResults: state.results.get(category) }))
|
||||||
|
.filter(category => category.categoryResults.length > 0)
|
||||||
|
.map(({ category, categoryResults }, i) => (
|
||||||
|
<Box key={i} width='max(50vw, 300px)' maxWidth='600px'>
|
||||||
|
<Rule borderTopWidth="0.5px" color="washedGray" />
|
||||||
|
<Text gray ml={2}>{category.charAt(0).toUpperCase() + category.slice(1)}</Text>
|
||||||
|
{categoryResults.map((result, i2) => (
|
||||||
|
<OmniboxResult
|
||||||
|
key={i2}
|
||||||
|
icon={result.app}
|
||||||
|
text={result.title}
|
||||||
|
subtext={cite(result.host)}
|
||||||
|
link={result.link}
|
||||||
|
navigate={() => this.navigate(result.link)}
|
||||||
|
selected={this.state.selected}
|
||||||
|
dark={props.dark} />
|
||||||
|
))}
|
||||||
|
</Box>
|
||||||
|
))
|
||||||
|
}
|
||||||
|
</Box>;
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
const { props, state } = this;
|
||||||
|
if (!state.selected && Array.from(this.state.results.values()).flat().length) {
|
||||||
|
this.setNextSelected();
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<Box
|
||||||
|
backgroundColor='scales.black30'
|
||||||
|
width='100vw'
|
||||||
|
height='100vh'
|
||||||
|
position='absolute'
|
||||||
|
top='0'
|
||||||
|
right='0'
|
||||||
|
zIndex='9'
|
||||||
|
display={props.show ? 'block' : 'none'}>
|
||||||
|
<Row justifyContent='center'>
|
||||||
|
<Box
|
||||||
|
mt='20vh'
|
||||||
|
width='max(50vw, 300px)'
|
||||||
|
maxWidth='600px'
|
||||||
|
borderRadius='2'
|
||||||
|
backgroundColor='white'
|
||||||
|
ref={(el) => {
|
||||||
|
this.omniBox = el;
|
||||||
|
}}>
|
||||||
|
<OmniboxInput
|
||||||
|
ref={(el) => {
|
||||||
|
this.omniInput = el;
|
||||||
|
}}
|
||||||
|
control={e => this.control(e)}
|
||||||
|
search={this.search}
|
||||||
|
query={state.query}
|
||||||
|
/>
|
||||||
|
{this.renderResults()}
|
||||||
|
</Box>
|
||||||
|
</Row>
|
||||||
|
</Box>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default withRouter(Omnibox);
|
26
pkg/interface/src/components/OmniboxInput.js
Normal file
26
pkg/interface/src/components/OmniboxInput.js
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
|
||||||
|
import React, { Component } from 'react';
|
||||||
|
|
||||||
|
export class OmniboxInput extends Component {
|
||||||
|
render() {
|
||||||
|
const { props } = this;
|
||||||
|
return (
|
||||||
|
<input
|
||||||
|
ref={(el) => {
|
||||||
|
this.input = el;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
className='ba b--transparent w-100 br2 white-d bg-gray0-d inter f9 pa2'
|
||||||
|
style={{ maxWidth: 'calc(600px - 1.15rem)', boxSizing: 'border-box' }}
|
||||||
|
placeholder='Search...'
|
||||||
|
onKeyDown={props.control}
|
||||||
|
onChange={props.search}
|
||||||
|
spellCheck={false}
|
||||||
|
value={props.query}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default OmniboxInput;
|
||||||
|
|
96
pkg/interface/src/components/OmniboxResult.js
Normal file
96
pkg/interface/src/components/OmniboxResult.js
Normal file
@ -0,0 +1,96 @@
|
|||||||
|
import React, { Component } from 'react';
|
||||||
|
import { Row, Icon, Text } from '@tlon/indigo-react';
|
||||||
|
import defaultApps from '../lib/default-apps';
|
||||||
|
|
||||||
|
export class OmniboxResult extends Component {
|
||||||
|
constructor(props) {
|
||||||
|
super(props);
|
||||||
|
this.state = {
|
||||||
|
isSelected: false,
|
||||||
|
hovered: false
|
||||||
|
};
|
||||||
|
this.setHover = this.setHover.bind(this);
|
||||||
|
this.result = React.createRef();
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidUpdate(prevProps) {
|
||||||
|
const { props, state } = this;
|
||||||
|
if (prevProps &&
|
||||||
|
!state.hovered &&
|
||||||
|
prevProps.selected !== props.selected &&
|
||||||
|
props.selected === props.link
|
||||||
|
) {
|
||||||
|
this.result.current.scrollIntoView({ block: 'nearest' });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setHover(boolean) {
|
||||||
|
this.setState({ hovered: boolean });
|
||||||
|
}
|
||||||
|
render() {
|
||||||
|
const { icon, text, subtext, link, navigate, selected, dark } = this.props;
|
||||||
|
|
||||||
|
let invertGraphic = {};
|
||||||
|
|
||||||
|
if (icon.toLowerCase() !== 'dojo') {
|
||||||
|
invertGraphic = (!dark && this.state.hovered) ||
|
||||||
|
selected === link ||
|
||||||
|
(dark && !(this.state.hovered || selected === link))
|
||||||
|
? { filter: 'invert(1)', paddingTop: 2 }
|
||||||
|
: { filter: 'invert(0)', paddingTop: 2 };
|
||||||
|
} else {
|
||||||
|
invertGraphic =
|
||||||
|
(!dark && this.state.hovered) ||
|
||||||
|
selected === link ||
|
||||||
|
(dark && !(this.state.hovered || selected === link))
|
||||||
|
? { filter: 'invert(0)', paddingTop: 2 }
|
||||||
|
: { filter: 'invert(1)', paddingTop: 2 };
|
||||||
|
}
|
||||||
|
|
||||||
|
let graphic = <div />;
|
||||||
|
if (defaultApps.includes(icon.toLowerCase()) || icon.toLowerCase() === 'links') {
|
||||||
|
graphic = <img className="mr2 v-mid" height="12" width="12" src={`/~landscape/img/${icon.toLowerCase()}.png`} style={invertGraphic} />;
|
||||||
|
} else {
|
||||||
|
graphic = <Icon verticalAlign="middle" mr={2} size="12px" />;
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<Row
|
||||||
|
py='2'
|
||||||
|
px='2'
|
||||||
|
display='flex'
|
||||||
|
flexDirection='row'
|
||||||
|
style={{ cursor: 'pointer' }}
|
||||||
|
onMouseEnter={() => this.setHover(true)}
|
||||||
|
onMouseLeave={() => this.setHover(false)}
|
||||||
|
backgroundColor={
|
||||||
|
this.state.hovered || selected === link ? 'blue' : 'white'
|
||||||
|
}
|
||||||
|
onClick={navigate}
|
||||||
|
width="100%"
|
||||||
|
ref={this.result}
|
||||||
|
>
|
||||||
|
{this.state.hovered || selected === link ? (
|
||||||
|
<>
|
||||||
|
{graphic}
|
||||||
|
<Text color='white' mr='1' style={{ 'flex-shrink': 0 }}>
|
||||||
|
{text}
|
||||||
|
</Text>
|
||||||
|
<Text pr='2' color='white' width='100%' textAlign='right'>
|
||||||
|
{subtext}
|
||||||
|
</Text>
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
{graphic}
|
||||||
|
<Text mr='1' style={{ 'flex-shrink': 0 }}>{text}</Text>
|
||||||
|
<Text pr='2' gray width='100%' textAlign='right'>
|
||||||
|
{subtext}
|
||||||
|
</Text>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</Row>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default OmniboxResult;
|
46
pkg/interface/src/components/ReconnectButton.js
Normal file
46
pkg/interface/src/components/ReconnectButton.js
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import { Box, Text } from '@tlon/indigo-react';
|
||||||
|
|
||||||
|
const ReconnectButton = ({ connection, subscription }) => {
|
||||||
|
const connectedStatus = connection || 'connected';
|
||||||
|
const reconnect = subscription.restart.bind(subscription);
|
||||||
|
if (connectedStatus === 'disconnected') {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Box
|
||||||
|
ml={4}
|
||||||
|
px={2}
|
||||||
|
py={1}
|
||||||
|
display='inline-block'
|
||||||
|
color='red'
|
||||||
|
border={1}
|
||||||
|
lineHeight='min'
|
||||||
|
borderRadius={2}
|
||||||
|
style={{ cursor: 'pointer' }}
|
||||||
|
onClick={reconnect}>
|
||||||
|
<Text color='red'>Reconnect ↻</Text>
|
||||||
|
</Box>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
} else if (connectedStatus === 'reconnecting') {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Box
|
||||||
|
ml={4}
|
||||||
|
px={2}
|
||||||
|
py={1}
|
||||||
|
lineHeight="min"
|
||||||
|
display='inline-block'
|
||||||
|
color='yellow'
|
||||||
|
border={1}
|
||||||
|
borderRadius={2}>
|
||||||
|
<Text color='yellow'>Reconnecting</Text>
|
||||||
|
</Box>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export default ReconnectButton;
|
@ -1,82 +1,103 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { useLocation, Link } from 'react-router-dom';
|
import { useLocation } from 'react-router-dom';
|
||||||
|
import { Box, Text, Icon } from '@tlon/indigo-react';
|
||||||
import GroupFilter from './GroupFilter';
|
import ReconnectButton from './ReconnectButton';
|
||||||
import { Sigil } from '../lib/sigil';
|
|
||||||
|
|
||||||
const getLocationName = (basePath) => {
|
|
||||||
if (basePath === '~chat')
|
|
||||||
return 'Chat';
|
|
||||||
else if (basePath === '~dojo')
|
|
||||||
return 'Dojo';
|
|
||||||
else if (basePath === '~groups')
|
|
||||||
return 'Groups';
|
|
||||||
else if (basePath === '~link')
|
|
||||||
return 'Links';
|
|
||||||
else if (basePath === '~publish')
|
|
||||||
return 'Publish';
|
|
||||||
else
|
|
||||||
return 'Unknown';
|
|
||||||
};
|
|
||||||
|
|
||||||
const StatusBar = (props) => {
|
const StatusBar = (props) => {
|
||||||
const location = useLocation();
|
const location = useLocation();
|
||||||
const basePath = location.pathname.split('/')[1];
|
const atHome = Boolean(location.pathname === '/');
|
||||||
const locationName = location.pathname === '/'
|
|
||||||
? 'Home'
|
|
||||||
: getLocationName(basePath);
|
|
||||||
|
|
||||||
const display = (!window.location.href.includes('popout/') &&
|
const display = (!window.location.href.includes('popout/'))
|
||||||
(locationName !== 'Unknown'))
|
|
||||||
? 'db' : 'dn';
|
? 'db' : 'dn';
|
||||||
|
|
||||||
const invites = (props.invites && props.invites['/contacts'])
|
const invites = (props.invites && props.invites['/contacts'])
|
||||||
? props.invites['/contacts']
|
? props.invites['/contacts']
|
||||||
: {};
|
: {};
|
||||||
const connection = props.connection || 'connected';
|
|
||||||
|
|
||||||
const reconnect = props.subscription.restart.bind(props.subscription);
|
const Notification = (Object.keys(invites).length > 0)
|
||||||
|
? <Icon size="22px" icon="Bullet"
|
||||||
|
fill="blue" position="absolute"
|
||||||
|
top={'-8px'} right={'7px'}
|
||||||
|
/>
|
||||||
|
: null;
|
||||||
|
|
||||||
|
const metaKey = (window.navigator.platform.includes('Mac')) ? '⌘' : 'Ctrl+';
|
||||||
|
|
||||||
|
const mobile = /Android|webOS|iPhone|iPad|iPod|BlackBerry/i.test(
|
||||||
|
navigator.userAgent
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className={
|
className={
|
||||||
'bg-white bg-gray0-d w-100 justify-between relative tc pt3 ' + display
|
'bg-white bg-gray0-d w-100 justify-between relative tc pt3 ' + display
|
||||||
}
|
}
|
||||||
style={{ height: 45 }}
|
style={{ height: 45 }}>
|
||||||
>
|
<div className='absolute left-0 pl4' style={{ top: 10 }}>
|
||||||
<div className="fl lh-copy absolute left-0 pl4" style={{ top: 8 }}>
|
{atHome ? null : (
|
||||||
<Link to="/~groups/me"
|
<Box
|
||||||
className="dib v-top" style={{ lineHeight: 0, paddingTop: 6 }}>
|
style={{ cursor: 'pointer' }}
|
||||||
<Sigil
|
display='inline-block'
|
||||||
ship={'~' + window.ship}
|
borderRadius={2}
|
||||||
classes="v-mid mix-blend-diff"
|
color='washedGray'
|
||||||
size={16}
|
border={1}
|
||||||
color={'#000000'}
|
py={1}
|
||||||
|
px={2}
|
||||||
|
mr={2}
|
||||||
|
onClick={() => props.history.push('/')}>
|
||||||
|
<img
|
||||||
|
className='invert-d'
|
||||||
|
src='/~landscape/img/icon-home.png'
|
||||||
|
height='12'
|
||||||
|
width='12'
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
)}
|
||||||
|
<Box
|
||||||
|
border={1}
|
||||||
|
borderRadius={2}
|
||||||
|
color='washedGray'
|
||||||
|
display='inline-block'
|
||||||
|
style={{ cursor: 'pointer' }}
|
||||||
|
lineHeight='min'
|
||||||
|
py={1}
|
||||||
|
px={2}
|
||||||
|
onClick={() => props.api.local.setOmnibox()}>
|
||||||
|
<Text display='inline-block' style={{ transform: 'rotate(180deg)' }}>
|
||||||
|
↩
|
||||||
|
</Text>
|
||||||
|
<Text ml={2} color='black'>
|
||||||
|
Leap
|
||||||
|
</Text>
|
||||||
|
<Text display={mobile ? 'none' : 'inline-block'} ml={4} color='gray'>
|
||||||
|
{metaKey}/
|
||||||
|
</Text>
|
||||||
|
</Box>
|
||||||
|
<ReconnectButton
|
||||||
|
connection={props.connection}
|
||||||
|
subscription={props.subscription}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className='fl absolute relative right-0 pr4' style={{ top: 10 }}>
|
||||||
|
<Box
|
||||||
|
style={{ cursor: 'pointer' }}
|
||||||
|
display='inline-block'
|
||||||
|
borderRadius={2}
|
||||||
|
color='washedGray'
|
||||||
|
lineHeight='min'
|
||||||
|
border={1}
|
||||||
|
px={2}
|
||||||
|
py={1}
|
||||||
|
onClick={() => props.history.push('/~groups')}>
|
||||||
|
<img
|
||||||
|
className='invert-d v-mid mr1'
|
||||||
|
src='/~landscape/img/groups.png'
|
||||||
|
height='16'
|
||||||
|
width='16'
|
||||||
/>
|
/>
|
||||||
</Link>
|
{Notification}
|
||||||
<GroupFilter invites={invites} associations={props.associations} api={props.api} />
|
<Text ml={1}>Groups</Text>
|
||||||
<span className="dib f9 v-mid gray2 ml1 mr1 c-default inter">/</span>
|
</Box>
|
||||||
{
|
|
||||||
location.pathname === '/'
|
|
||||||
? null
|
|
||||||
: <Link
|
|
||||||
className="dib f9 v-mid inter ml2 no-underline white-d"
|
|
||||||
to="/"
|
|
||||||
style={{ top: 14 }}
|
|
||||||
>
|
|
||||||
⟵
|
|
||||||
</Link>
|
|
||||||
}
|
|
||||||
<p className="dib f9 v-mid inter ml2 white-d">{locationName}</p>
|
|
||||||
{ connection === 'disconnected' &&
|
|
||||||
(<span
|
|
||||||
onClick={reconnect}
|
|
||||||
className="ml4 ph2 dib f9 v-mid red2 inter ba b-red2 br1 pointer"
|
|
||||||
>Reconnect ↻</span> )
|
|
||||||
}
|
|
||||||
{ connection === 'reconnecting' &&
|
|
||||||
(<span className="ml4 ph2 dib f9 v-mid yellow2 inter ba b-yellow2 br1">Reconnecting</span> )
|
|
||||||
}
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
3
pkg/interface/src/lib/default-apps.js
Normal file
3
pkg/interface/src/lib/default-apps.js
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
const defaultApps = ['chat', 'dojo', 'groups', 'link', 'publish'];
|
||||||
|
|
||||||
|
export default defaultApps;
|
120
pkg/interface/src/lib/omnibox.js
Normal file
120
pkg/interface/src/lib/omnibox.js
Normal file
@ -0,0 +1,120 @@
|
|||||||
|
import defaultApps from './default-apps';
|
||||||
|
|
||||||
|
const indexes = new Map([
|
||||||
|
['commands', []],
|
||||||
|
['subscriptions', []],
|
||||||
|
['groups', []],
|
||||||
|
['apps', []]
|
||||||
|
]);
|
||||||
|
|
||||||
|
// result schematic
|
||||||
|
const result = function(title, link, app, host) {
|
||||||
|
return {
|
||||||
|
'title': title,
|
||||||
|
'link': link,
|
||||||
|
'app': app,
|
||||||
|
'host': host
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
const commandIndex = function () {
|
||||||
|
// commands are special cased for default suite
|
||||||
|
const commands = [];
|
||||||
|
defaultApps
|
||||||
|
.filter((e) => {
|
||||||
|
return e !== 'dojo';
|
||||||
|
})
|
||||||
|
.map((e) => {
|
||||||
|
let title = e;
|
||||||
|
if (e === 'link') {
|
||||||
|
title = 'Links';
|
||||||
|
}
|
||||||
|
|
||||||
|
title = title.charAt(0).toUpperCase() + title.slice(1);
|
||||||
|
|
||||||
|
let obj = result(`${title}: Create`, `/~${e}/new`, e, null);
|
||||||
|
commands.push(obj);
|
||||||
|
|
||||||
|
if (title === 'Groups') {
|
||||||
|
obj = result(`${title}: Join Group`, `/~${e}/join`, title, null);
|
||||||
|
commands.push(obj);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return commands;
|
||||||
|
};
|
||||||
|
|
||||||
|
const appIndex = function (apps) {
|
||||||
|
// all apps are indexed from launch data
|
||||||
|
// indexed into 'apps'
|
||||||
|
const applications = [];
|
||||||
|
Object.keys(apps)
|
||||||
|
.filter((e) => {
|
||||||
|
return apps[e]?.type?.basic;
|
||||||
|
})
|
||||||
|
.map((e) => {
|
||||||
|
const obj = result(
|
||||||
|
apps[e].type.basic.title,
|
||||||
|
apps[e].type.basic.linkedUrl,
|
||||||
|
apps[e].type.basic.title,
|
||||||
|
null
|
||||||
|
);
|
||||||
|
applications.push(obj);
|
||||||
|
});
|
||||||
|
// add groups separately
|
||||||
|
applications.push(
|
||||||
|
result('Groups', '/~groups', 'groups', null)
|
||||||
|
);
|
||||||
|
return applications;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function index(associations, apps) {
|
||||||
|
// all metadata from all apps is indexed
|
||||||
|
// into subscriptions and groups
|
||||||
|
const subscriptions = [];
|
||||||
|
const groups = [];
|
||||||
|
Object.keys(associations).filter((e) => {
|
||||||
|
// skip apps with no metadata
|
||||||
|
return Object.keys(associations[e]).length > 0;
|
||||||
|
}).map((e) => {
|
||||||
|
// iterate through each app's metadata object
|
||||||
|
Object.keys(associations[e]).map((association) => {
|
||||||
|
const each = associations[e][association];
|
||||||
|
let title = each['app-path'];
|
||||||
|
if (each.metadata.title !== '') {
|
||||||
|
title = each.metadata.title;
|
||||||
|
}
|
||||||
|
|
||||||
|
let app = each['app-name'];
|
||||||
|
if (each['app-name'] === 'contacts') {
|
||||||
|
app = 'groups';
|
||||||
|
};
|
||||||
|
|
||||||
|
const shipStart = each['app-path'].substr(each['app-path'].indexOf('~'));
|
||||||
|
|
||||||
|
if (app === 'groups') {
|
||||||
|
const obj = result(
|
||||||
|
title,
|
||||||
|
`/~${app}${each['app-path']}`,
|
||||||
|
app.charAt(0).toUpperCase() + app.slice(1),
|
||||||
|
shipStart.slice(0, shipStart.indexOf('/'))
|
||||||
|
);
|
||||||
|
groups.push(obj);
|
||||||
|
} else {
|
||||||
|
const obj = result(
|
||||||
|
title,
|
||||||
|
`/~${each['app-name']}/join${each['app-path']}`,
|
||||||
|
app.charAt(0).toUpperCase() + app.slice(1),
|
||||||
|
shipStart.slice(0, shipStart.indexOf('/'))
|
||||||
|
);
|
||||||
|
subscriptions.push(obj);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
indexes.set('commands', commandIndex());
|
||||||
|
indexes.set('subscriptions', subscriptions);
|
||||||
|
indexes.set('groups', groups);
|
||||||
|
indexes.set('apps', appIndex(apps));
|
||||||
|
|
||||||
|
return indexes;
|
||||||
|
};
|
@ -2,24 +2,24 @@ import React, { Component } from 'react';
|
|||||||
import { sigil, reactRenderer } from 'urbit-sigil-js';
|
import { sigil, reactRenderer } from 'urbit-sigil-js';
|
||||||
|
|
||||||
export class Sigil extends Component {
|
export class Sigil extends Component {
|
||||||
|
static foregroundFromBackground(background) {
|
||||||
|
const rgb = {
|
||||||
|
r: parseInt(background.slice(1, 3), 16),
|
||||||
|
g: parseInt(background.slice(3, 5), 16),
|
||||||
|
b: parseInt(background.slice(5, 7), 16)
|
||||||
|
};
|
||||||
|
const brightness = ((299 * rgb.r) + (587 * rgb.g) + (114 * rgb.b)) / 1000;
|
||||||
|
const whiteBrightness = 255;
|
||||||
|
|
||||||
|
return ((whiteBrightness - brightness) < 50) ? 'black' : 'white';
|
||||||
|
}
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
const { props } = this;
|
const { props } = this;
|
||||||
|
|
||||||
const classes = props.classes || '';
|
const classes = props.classes || '';
|
||||||
|
|
||||||
const rgb = {
|
const foreground = Sigil.foregroundFromBackground(props.color);
|
||||||
r: parseInt(props.color.slice(1, 3), 16),
|
|
||||||
g: parseInt(props.color.slice(3, 5), 16),
|
|
||||||
b: parseInt(props.color.slice(5, 7), 16)
|
|
||||||
};
|
|
||||||
const brightness = ((299 * rgb.r) + (587 * rgb.g) + (114 * rgb.b)) / 1000;
|
|
||||||
const whiteBrightness = 255;
|
|
||||||
|
|
||||||
let foreground = 'white';
|
|
||||||
|
|
||||||
if ((whiteBrightness - brightness) < 50) {
|
|
||||||
foreground = 'black';
|
|
||||||
}
|
|
||||||
|
|
||||||
if (props.ship.length > 14) {
|
if (props.ship.length > 14) {
|
||||||
return (
|
return (
|
||||||
|
@ -119,6 +119,9 @@ export function writeText(str) {
|
|||||||
// trim patps to match dojo, chat-cli
|
// trim patps to match dojo, chat-cli
|
||||||
export function cite(ship) {
|
export function cite(ship) {
|
||||||
let patp = ship, shortened = '';
|
let patp = ship, shortened = '';
|
||||||
|
if (patp === null || patp === '') {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
if (patp.startsWith('~')) {
|
if (patp.startsWith('~')) {
|
||||||
patp = patp.substr(1);
|
patp = patp.substr(1);
|
||||||
}
|
}
|
||||||
@ -275,3 +278,38 @@ export function stringToSymbol(str) {
|
|||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function scrollIsAtTop(container) {
|
||||||
|
if (
|
||||||
|
(navigator.userAgent.includes("Safari") &&
|
||||||
|
navigator.userAgent.includes("Chrome")) ||
|
||||||
|
navigator.userAgent.includes("Firefox")
|
||||||
|
) {
|
||||||
|
return container.scrollTop === 0;
|
||||||
|
} else if (navigator.userAgent.includes("Safari")) {
|
||||||
|
return (
|
||||||
|
container.scrollHeight + Math.round(container.scrollTop) <=
|
||||||
|
container.clientHeight + 10
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function scrollIsAtBottom(container) {
|
||||||
|
if (
|
||||||
|
(navigator.userAgent.includes("Safari") &&
|
||||||
|
navigator.userAgent.includes("Chrome")) ||
|
||||||
|
navigator.userAgent.includes("Firefox")
|
||||||
|
) {
|
||||||
|
return (
|
||||||
|
container.scrollHeight - Math.round(container.scrollTop) <=
|
||||||
|
container.clientHeight + 10
|
||||||
|
);
|
||||||
|
} else if (navigator.userAgent.includes("Safari")) {
|
||||||
|
return container.scrollTop === 0;
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@ -50,7 +50,6 @@ export default class LaunchReducer<S extends LaunchState> {
|
|||||||
|
|
||||||
changeIsShown(json: LaunchUpdate, state: S) {
|
changeIsShown(json: LaunchUpdate, state: S) {
|
||||||
const data = _.get(json, 'changeIsShown', false);
|
const data = _.get(json, 'changeIsShown', false);
|
||||||
console.log(json, data);
|
|
||||||
if (data) {
|
if (data) {
|
||||||
let tile = state.launch.tiles[data.name];
|
let tile = state.launch.tiles[data.name];
|
||||||
console.log(tile);
|
console.log(tile);
|
||||||
|
@ -3,16 +3,16 @@ import { StoreState } from '../store/type';
|
|||||||
import { Cage } from '../types/cage';
|
import { Cage } from '../types/cage';
|
||||||
import { LocalUpdate } from '../types/local-update';
|
import { LocalUpdate } from '../types/local-update';
|
||||||
|
|
||||||
type LocalState = Pick<StoreState, 'sidebarShown' | 'selectedGroups' | 'dark' | 'baseHash'>;
|
type LocalState = Pick<StoreState, 'sidebarShown' | 'omniboxShown' | 'dark' | 'baseHash'>;
|
||||||
|
|
||||||
export default class LocalReducer<S extends LocalState> {
|
export default class LocalReducer<S extends LocalState> {
|
||||||
reduce(json: Cage, state: S) {
|
reduce(json: Cage, state: S) {
|
||||||
const data = json['local'];
|
const data = json['local'];
|
||||||
if (data) {
|
if (data) {
|
||||||
this.sidebarToggle(data, state);
|
this.sidebarToggle(data, state);
|
||||||
this.setSelected(data, state);
|
|
||||||
this.setDark(data, state);
|
this.setDark(data, state);
|
||||||
this.baseHash(data, state);
|
this.baseHash(data, state);
|
||||||
|
this.omniboxShown(data, state);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
baseHash(obj: LocalUpdate, state: S) {
|
baseHash(obj: LocalUpdate, state: S) {
|
||||||
@ -21,18 +21,18 @@ export default class LocalReducer<S extends LocalState> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
omniboxShown(obj: LocalUpdate, state: S) {
|
||||||
|
if ('omniboxShown' in obj) {
|
||||||
|
state.omniboxShown = !state.omniboxShown;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
sidebarToggle(obj: LocalUpdate, state: S) {
|
sidebarToggle(obj: LocalUpdate, state: S) {
|
||||||
if ('sidebarToggle' in obj) {
|
if ('sidebarToggle' in obj) {
|
||||||
state.sidebarShown = !state.sidebarShown;
|
state.sidebarShown = !state.sidebarShown;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
setSelected(obj: LocalUpdate, state: S) {
|
|
||||||
if ('selected' in obj) {
|
|
||||||
state.selectedGroups = obj.selected;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
setDark(obj: LocalUpdate, state: S) {
|
setDark(obj: LocalUpdate, state: S) {
|
||||||
if('setDark' in obj) {
|
if('setDark' in obj) {
|
||||||
state.dark = obj.setDark;
|
state.dark = obj.setDark;
|
||||||
|
@ -41,6 +41,7 @@ export default class GlobalStore extends BaseStore<StoreState> {
|
|||||||
chatInitialized: false,
|
chatInitialized: false,
|
||||||
connection: 'connected',
|
connection: 'connected',
|
||||||
sidebarShown: true,
|
sidebarShown: true,
|
||||||
|
omniboxShown: false,
|
||||||
baseHash: null,
|
baseHash: null,
|
||||||
invites: {},
|
invites: {},
|
||||||
associations: {
|
associations: {
|
||||||
@ -72,7 +73,6 @@ export default class GlobalStore extends BaseStore<StoreState> {
|
|||||||
linkComments: {},
|
linkComments: {},
|
||||||
notebooks: {},
|
notebooks: {},
|
||||||
contacts: {},
|
contacts: {},
|
||||||
selectedGroups: [],
|
|
||||||
dark: false,
|
dark: false,
|
||||||
inbox: {},
|
inbox: {},
|
||||||
chatSynced: null,
|
chatSynced: null,
|
||||||
|
@ -2,7 +2,6 @@ import { Inbox, Envelope } from '../types/chat-update';
|
|||||||
import { ChatHookUpdate } from '../types/chat-hook-update';
|
import { ChatHookUpdate } from '../types/chat-hook-update';
|
||||||
import { Path } from '../types/noun';
|
import { Path } from '../types/noun';
|
||||||
import { Invites } from '../types/invite-update';
|
import { Invites } from '../types/invite-update';
|
||||||
import { SelectedGroup } from '../types/local-update';
|
|
||||||
import { Associations } from '../types/metadata-update';
|
import { Associations } from '../types/metadata-update';
|
||||||
import { Rolodex } from '../types/contact-update';
|
import { Rolodex } from '../types/contact-update';
|
||||||
import { Notebooks } from '../types/publish-update';
|
import { Notebooks } from '../types/publish-update';
|
||||||
@ -16,7 +15,7 @@ import { ConnectionStatus } from '../types/connection';
|
|||||||
export interface StoreState {
|
export interface StoreState {
|
||||||
// local state
|
// local state
|
||||||
sidebarShown: boolean;
|
sidebarShown: boolean;
|
||||||
selectedGroups: SelectedGroup[];
|
omniboxShown: boolean;
|
||||||
dark: boolean;
|
dark: boolean;
|
||||||
connection: ConnectionStatus;
|
connection: ConnectionStatus;
|
||||||
baseHash: string | null;
|
baseHash: string | null;
|
||||||
|
@ -1,19 +1,13 @@
|
|||||||
import { Path } from './noun';
|
|
||||||
|
|
||||||
export type LocalUpdate =
|
export type LocalUpdate =
|
||||||
LocalUpdateSidebarToggle
|
LocalUpdateSidebarToggle
|
||||||
| LocalUpdateSelectedGroups
|
|
||||||
| LocalUpdateSetDark
|
| LocalUpdateSetDark
|
||||||
|
| LocalUpdateSetOmniboxShown
|
||||||
| LocalUpdateBaseHash;
|
| LocalUpdateBaseHash;
|
||||||
|
|
||||||
interface LocalUpdateSidebarToggle {
|
interface LocalUpdateSidebarToggle {
|
||||||
sidebarToggle: boolean;
|
sidebarToggle: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface LocalUpdateSelectedGroups {
|
|
||||||
selected: SelectedGroup[];
|
|
||||||
}
|
|
||||||
|
|
||||||
interface LocalUpdateSetDark {
|
interface LocalUpdateSetDark {
|
||||||
setDark: boolean;
|
setDark: boolean;
|
||||||
}
|
}
|
||||||
@ -22,4 +16,6 @@ interface LocalUpdateBaseHash {
|
|||||||
baseHash: string;
|
baseHash: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export type SelectedGroup = [Path, string];
|
interface LocalUpdateSetOmniboxShown {
|
||||||
|
omniboxShown: boolean;
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user