Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
C
converse.js
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
converse.js
Commits
dd5f6f10
Commit
dd5f6f10
authored
May 24, 2018
by
JC Brand
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Show join/leave notifications for members who go offline
updates #1094
parent
04e95a17
Changes
3
Show whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
91 additions
and
14 deletions
+91
-14
spec/chatroom.js
spec/chatroom.js
+67
-7
src/converse-muc-views.js
src/converse-muc-views.js
+16
-3
src/converse-muc.js
src/converse-muc.js
+8
-4
No files found.
spec/chatroom.js
View file @
dd5f6f10
...
...
@@ -657,6 +657,7 @@
expect
(
indicator
.
getAttribute
(
'
data-isodate
'
)).
toEqual
(
moment
().
startOf
(
'
day
'
).
format
());
expect
(
indicator
.
querySelector
(
'
time
'
).
getAttribute
(
'
class
'
)).
toEqual
(
'
separator-text
'
);
expect
(
indicator
.
querySelector
(
'
time
'
).
textContent
).
toEqual
(
moment
().
startOf
(
'
day
'
).
format
(
"
dddd MMM Do YYYY
"
));
expect
(
chat_content
.
querySelectorAll
(
'
div.chat-info
'
).
length
).
toBe
(
2
);
expect
(
chat_content
.
querySelector
(
'
div.chat-info:last-child
'
).
textContent
).
toBe
(
"
some1 has entered the room
"
);
...
...
@@ -672,9 +673,9 @@
.
c
(
'
status
'
,
'
Disconnected: Replaced by new connection
'
).
up
()
.
c
(
'
x
'
,
{
xmlns
:
Strophe
.
NS
.
MUC_USER
})
.
c
(
'
item
'
,
{
'
affiliation
'
:
'
none
'
,
'
affiliation
'
:
'
owner
'
,
'
jid
'
:
'
some1@localhost/_converse.js-290929789
'
,
'
role
'
:
'
none
'
'
role
'
:
'
moderator
'
});
_converse
.
connection
.
_dataRecv
(
test_utils
.
createRequest
(
presence
));
...
...
@@ -685,7 +686,7 @@
expect
(
indicator
.
getAttribute
(
'
data-isodate
'
)).
toEqual
(
moment
().
startOf
(
'
day
'
).
format
());
expect
(
indicator
.
querySelector
(
'
time
'
).
textContent
).
toEqual
(
moment
().
startOf
(
'
day
'
).
format
(
"
dddd MMM Do YYYY
"
));
expect
(
$
(
chat_content
).
find
(
'
div.chat-info
'
).
length
).
toBe
(
4
);
expect
(
chat_content
.
querySelectorAll
(
'
div.chat-info
'
).
length
).
toBe
(
3
);
expect
(
$
(
chat_content
).
find
(
'
div.chat-info:last
'
).
html
()).
toBe
(
'
some1 has left the room.
'
+
'
"Disconnected: Replaced by new connection"
'
);
...
...
@@ -720,7 +721,7 @@
expect
(
$indicator
.
attr
(
'
class
'
)).
toEqual
(
'
message date-separator
'
);
expect
(
$indicator
.
data
(
'
isodate
'
)).
toEqual
(
moment
().
startOf
(
'
day
'
).
format
());
expect
(
$indicator
.
find
(
'
time
'
).
text
()).
toEqual
(
moment
().
startOf
(
'
day
'
).
format
(
"
dddd MMM Do YYYY
"
));
expect
(
$chat_content
.
find
(
'
div.chat-info
'
).
length
).
toBe
(
5
);
expect
(
chat_content
.
querySelectorAll
(
'
div.chat-info
'
).
length
).
toBe
(
4
);
expect
(
$chat_content
.
find
(
'
div.chat-info:last
'
).
html
()).
toBe
(
"
newguy has entered the room
"
);
jasmine
.
clock
().
tick
(
ONE_DAY_LATER
);
...
...
@@ -760,7 +761,7 @@
expect
(
$indicator
.
data
(
'
isodate
'
)).
toEqual
(
moment
().
startOf
(
'
day
'
).
format
());
expect
(
$indicator
.
find
(
'
time
'
).
text
()).
toEqual
(
moment
().
startOf
(
'
day
'
).
format
(
"
dddd MMM Do YYYY
"
));
expect
(
$chat_content
.
find
(
'
div.chat-info
'
).
length
).
toBe
(
6
);
expect
(
chat_content
.
querySelectorAll
(
'
div.chat-info
'
).
length
).
toBe
(
5
);
expect
(
$chat_content
.
find
(
'
div.chat-info:last
'
).
html
()).
toBe
(
'
newguy has left the room.
'
+
'
"Disconnected: Replaced by new connection"
'
);
...
...
@@ -1113,7 +1114,7 @@
}).
catch
(
_
.
partial
(
_converse
.
log
,
_
,
Strophe
.
LogLevel
.
FATAL
));
}));
it
(
"
shows
users
currently present in the room
"
,
it
(
"
shows
all members even if they're not
currently present in the room
"
,
mock
.
initConverseWithPromises
(
null
,
[
'
rosterGroupsFetched
'
],
{},
function
(
done
,
_converse
)
{
...
...
@@ -1145,6 +1146,61 @@
expect
(
occupants
.
querySelectorAll
(
'
li .occupant-nick
'
)[
index
].
textContent
.
trim
()).
toBe
(
mock
.
chatroom_names
[
i
]);
}
// Test users leaving the room
// http://xmpp.org/extensions/xep-0045.html#exit
for
(
i
=
mock
.
chatroom_names
.
length
-
1
;
i
>-
1
;
i
--
)
{
name
=
mock
.
chatroom_names
[
i
];
role
=
mock
.
chatroom_roles
[
name
].
role
;
// See example 21 http://xmpp.org/extensions/xep-0045.html#enter-pres
presence
=
$pres
({
to
:
'
dummy@localhost/pda
'
,
from
:
'
lounge@localhost/
'
+
name
,
type
:
'
unavailable
'
}).
c
(
'
x
'
).
attrs
({
xmlns
:
'
http://jabber.org/protocol/muc#user
'
})
.
c
(
'
item
'
).
attrs
({
affiliation
:
mock
.
chatroom_roles
[
name
].
affiliation
,
jid
:
name
.
replace
(
/ /g
,
'
.
'
).
toLowerCase
()
+
'
@localhost
'
,
role
:
'
none
'
}).
nodeTree
;
_converse
.
connection
.
_dataRecv
(
test_utils
.
createRequest
(
presence
));
expect
(
occupants
.
querySelectorAll
(
'
li
'
).
length
).
toBe
(
7
);
}
done
();
}).
catch
(
_
.
partial
(
console
.
error
,
_
));
}));
it
(
"
shows users currently present in the room
"
,
mock
.
initConverseWithPromises
(
null
,
[
'
rosterGroupsFetched
'
],
{},
function
(
done
,
_converse
)
{
test_utils
.
openAndEnterChatRoom
(
_converse
,
'
lounge
'
,
'
localhost
'
,
'
dummy
'
).
then
(
function
()
{
var
name
;
var
view
=
_converse
.
chatboxviews
.
get
(
'
lounge@localhost
'
),
occupants
=
view
.
el
.
querySelector
(
'
.occupant-list
'
);
var
presence
,
role
,
jid
,
model
;
for
(
var
i
=
0
;
i
<
mock
.
chatroom_names
.
length
;
i
++
)
{
name
=
mock
.
chatroom_names
[
i
];
role
=
mock
.
chatroom_roles
[
name
].
role
;
// See example 21 http://xmpp.org/extensions/xep-0045.html#enter-pres
jid
=
presence
=
$pres
({
to
:
'
dummy@localhost/pda
'
,
from
:
'
lounge@localhost/
'
+
name
}).
c
(
'
x
'
).
attrs
({
xmlns
:
'
http://jabber.org/protocol/muc#user
'
})
.
c
(
'
item
'
).
attrs
({
affiliation
:
'
none
'
,
jid
:
name
.
replace
(
/ /g
,
'
.
'
).
toLowerCase
()
+
'
@localhost
'
,
role
:
role
}).
up
()
.
c
(
'
status
'
).
attrs
({
code
:
'
110
'
}).
nodeTree
;
_converse
.
connection
.
_dataRecv
(
test_utils
.
createRequest
(
presence
));
expect
(
occupants
.
querySelectorAll
(
'
li
'
).
length
).
toBe
(
2
+
i
);
model
=
view
.
occupantsview
.
model
.
where
({
'
nick
'
:
name
})[
0
];
var
index
=
view
.
occupantsview
.
model
.
indexOf
(
model
);
expect
(
occupants
.
querySelectorAll
(
'
li .occupant-nick
'
)[
index
].
textContent
.
trim
()).
toBe
(
mock
.
chatroom_names
[
i
]);
}
// Test users leaving the room
// http://xmpp.org/extensions/xep-0045.html#exit
for
(
i
=
mock
.
chatroom_names
.
length
-
1
;
i
>-
1
;
i
--
)
{
...
...
@@ -1710,7 +1766,11 @@
.
c
(
'
status
'
).
attrs
({
code
:
'
110
'
}).
nodeTree
;
_converse
.
connection
.
_dataRecv
(
test_utils
.
createRequest
(
presence
));
expect
(
$chat_content
.
find
(
'
div.chat-info
'
).
length
).
toBe
(
2
);
// XXX: currently we still have an additional "has entered the room"
// notification for the new nickname. Ideally we'd not have
// that, but that's probably not possible without some
// significant refactoring.
expect
(
$chat_content
.
find
(
'
div.chat-info
'
).
length
).
toBe
(
3
);
expect
(
$chat_content
.
find
(
'
div.chat-info
'
).
get
(
1
).
textContent
).
toBe
(
__
(
_converse
.
muc
.
new_nickname_messages
[
"
303
"
],
"
newnick
"
)
);
...
...
src/converse-muc-views.js
View file @
dd5f6f10
...
...
@@ -523,6 +523,16 @@
this
.
model
.
occupants
.
on
(
'
add
'
,
this
.
showJoinNotification
,
this
);
this
.
model
.
occupants
.
on
(
'
remove
'
,
this
.
showLeaveNotification
,
this
);
this
.
model
.
occupants
.
on
(
'
change:show
'
,
(
occupant
)
=>
{
if
(
!
occupant
.
isMember
()
||
_
.
includes
(
occupant
.
get
(
'
states
'
),
'
303
'
))
{
return
;
}
if
(
occupant
.
get
(
'
show
'
)
===
'
offline
'
)
{
this
.
showLeaveNotification
(
occupant
);
}
else
if
(
occupant
.
get
(
'
show
'
)
===
'
online
'
)
{
this
.
showJoinNotification
(
occupant
);
}
});
this
.
createEmojiPicker
();
this
.
createOccupantsView
();
...
...
@@ -1354,10 +1364,13 @@
},
showLeaveNotification
(
occupant
)
{
const
nick
=
occupant
.
get
(
'
nick
'
);
const
stat
=
occupant
.
get
(
'
status
'
);
const
last_el
=
this
.
content
.
lastElementChild
;
const
nick
=
occupant
.
get
(
'
nick
'
),
stat
=
occupant
.
get
(
'
status
'
),
last_el
=
this
.
content
.
lastElementChild
,
last_msg_date
=
last_el
.
getAttribute
(
'
data-isodate
'
);
if
(
_
.
includes
(
_
.
get
(
last_el
,
'
classList
'
,
[]),
'
chat-info
'
)
&&
moment
(
last_msg_date
).
isSame
(
new
Date
(),
"
day
"
)
&&
_
.
get
(
last_el
,
'
dataset
'
,
{}).
join
===
`"
${
nick
}
"`
)
{
let
message
;
...
...
src/converse-muc.js
View file @
dd5f6f10
...
...
@@ -742,8 +742,7 @@
}
const
occupant
=
this
.
occupants
.
findOccupant
(
data
);
if
(
data
.
type
===
'
unavailable
'
&&
occupant
)
{
if
(
!
_
.
includes
(
data
.
states
,
converse
.
MUC_NICK_CHANGED_CODE
)
&&
!
_
.
includes
([
'
admin
'
,
'
owner
'
,
'
member
'
],
occupant
.
get
(
'
affiliation
'
)))
{
if
(
!
_
.
includes
(
data
.
states
,
converse
.
MUC_NICK_CHANGED_CODE
)
&&
!
occupant
.
isMember
())
{
// We only destroy the occupant if this is not a nickname change operation.
// and if they're not on the member lists.
// Before destroying we set the new data, so
...
...
@@ -767,12 +766,13 @@
parsePresence
(
pres
)
{
const
from
=
pres
.
getAttribute
(
"
from
"
),
type
=
pres
.
getAttribute
(
"
type
"
),
data
=
{
'
from
'
:
from
,
'
nick
'
:
Strophe
.
getResourceFromJid
(
from
),
'
type
'
:
pres
.
getAttribute
(
"
type
"
)
,
'
type
'
:
type
,
'
states
'
:
[],
'
show
'
:
'
on
line
'
'
show
'
:
type
!==
'
unavailable
'
?
'
online
'
:
'
off
line
'
};
_
.
each
(
pres
.
childNodes
,
function
(
child
)
{
switch
(
child
.
nodeName
)
{
...
...
@@ -996,6 +996,10 @@
getDisplayName
()
{
return
this
.
get
(
'
nick
'
)
||
this
.
get
(
'
jid
'
);
},
isMember
()
{
return
_
.
includes
([
'
admin
'
,
'
owner
'
,
'
member
'
],
this
.
get
(
'
affiliation
'
));
}
});
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment