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
7ad39cfd
Commit
7ad39cfd
authored
Dec 05, 2016
by
JC Brand
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Fetch the room information before opening the room.
parent
67cdf5da
Changes
6
Hide whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
237 additions
and
95 deletions
+237
-95
docs/CHANGES.md
docs/CHANGES.md
+2
-3
spec/chatroom.js
spec/chatroom.js
+140
-33
src/converse-bookmarks.js
src/converse-bookmarks.js
+1
-1
src/converse-chatview.js
src/converse-chatview.js
+0
-1
src/converse-muc.js
src/converse-muc.js
+84
-57
tests/utils.js
tests/utils.js
+10
-0
No files found.
docs/CHANGES.md
View file @
7ad39cfd
...
@@ -2,9 +2,8 @@
...
@@ -2,9 +2,8 @@
## 2.0.4 (Unreleased)
## 2.0.4 (Unreleased)
-
#737: Bugfix. Translations weren't being applied. [jcbrand]
-
#737: Bugfix. Translations weren't being applied. [jcbrand]
-
Room information as returned by the server is now stored on the room model.
-
Fetch room info and store it on the room model.
For context, see: http://xmpp.org/extensions/xep-0045.html#disco-roominfo
For context, see: http://xmpp.org/extensions/xep-0045.html#disco-roominfo [jcbrand]
[jcbrand]
-
Bugfix. Switching from bookmarks form to config form shows only the spinner. [jcbrand]
-
Bugfix. Switching from bookmarks form to config form shows only the spinner. [jcbrand]
-
Bugfix. Other room occupants sometimes not shown when reloading the page. [jcbrand]
-
Bugfix. Other room occupants sometimes not shown when reloading the page. [jcbrand]
-
Bugfix. Due to changes in
`converse-core`
the controlbox wasn't aware anymore of
-
Bugfix. Due to changes in
`converse-core`
the controlbox wasn't aware anymore of
...
...
spec/chatroom.js
View file @
7ad39cfd
...
@@ -160,6 +160,17 @@
...
@@ -160,6 +160,17 @@
'
whois
'
:
'
anyone
'
'
whois
'
:
'
anyone
'
}
}
});
});
// We pretend this is a new room, so no disco info is returned.
var
features_stanza
=
$iq
({
from
:
'
room@conference.example.org
'
,
'
id
'
:
IQ_id
,
'
to
'
:
'
dummy@localhost/desktop
'
,
'
type
'
:
'
error
'
}).
c
(
'
error
'
,
{
'
type
'
:
'
cancel
'
})
.
c
(
'
item-not-found
'
,
{
'
xmlns
'
:
"
urn:ietf:params:xml:ns:xmpp-stanzas
"
});
converse
.
connection
.
_dataRecv
(
test_utils
.
createRequest
(
features_stanza
));
/* <presence xmlns="jabber:client" to="dummy@localhost/pda" from="room@conference.example.org/yo">
/* <presence xmlns="jabber:client" to="dummy@localhost/pda" from="room@conference.example.org/yo">
* <x xmlns="http://jabber.org/protocol/muc#user">
* <x xmlns="http://jabber.org/protocol/muc#user">
* <item affiliation="owner" jid="dummy@localhost/pda" role="moderator"/>
* <item affiliation="owner" jid="dummy@localhost/pda" role="moderator"/>
...
@@ -251,11 +262,20 @@
...
@@ -251,11 +262,20 @@
sent_IQ
=
iq
;
sent_IQ
=
iq
;
IQ_id
=
sendIQ
.
bind
(
this
)(
iq
,
callback
,
errback
);
IQ_id
=
sendIQ
.
bind
(
this
)(
iq
,
callback
,
errback
);
});
});
runs
(
function
()
{
runs
(
function
()
{
converse_api
.
rooms
.
open
(
'
coven@chat.shakespeare.lit
'
,
{
'
nick
'
:
'
some1
'
});
converse_api
.
rooms
.
open
(
'
coven@chat.shakespeare.lit
'
,
{
'
nick
'
:
'
some1
'
});
view
=
converse
.
chatboxviews
.
get
(
'
coven@chat.shakespeare.lit
'
);
view
=
converse
.
chatboxviews
.
get
(
'
coven@chat.shakespeare.lit
'
);
spyOn
(
view
,
'
findAndSaveOwnAffiliation
'
).
andCallThrough
();
spyOn
(
view
,
'
saveAffiliationAndRole
'
).
andCallThrough
();
// We pretend this is a new room, so no disco info is returned.
var
features_stanza
=
$iq
({
from
:
'
coven@chat.shakespeare.lit
'
,
'
id
'
:
IQ_id
,
'
to
'
:
'
dummy@localhost/desktop
'
,
'
type
'
:
'
error
'
}).
c
(
'
error
'
,
{
'
type
'
:
'
cancel
'
})
.
c
(
'
item-not-found
'
,
{
'
xmlns
'
:
"
urn:ietf:params:xml:ns:xmpp-stanzas
"
});
converse
.
connection
.
_dataRecv
(
test_utils
.
createRequest
(
features_stanza
));
/* <presence to="dummy@localhost/converse.js-29092160"
/* <presence to="dummy@localhost/converse.js-29092160"
* from="coven@chat.shakespeare.lit/some1">
* from="coven@chat.shakespeare.lit/some1">
...
@@ -276,7 +296,7 @@
...
@@ -276,7 +296,7 @@
}).
up
()
}).
up
()
.
c
(
'
status
'
,
{
code
:
'
110
'
});
.
c
(
'
status
'
,
{
code
:
'
110
'
});
converse
.
connection
.
_dataRecv
(
test_utils
.
createRequest
(
presence
));
converse
.
connection
.
_dataRecv
(
test_utils
.
createRequest
(
presence
));
expect
(
view
.
findAndSaveOwnAffiliation
).
toHaveBeenCalled
();
expect
(
view
.
saveAffiliationAndRole
).
toHaveBeenCalled
();
expect
(
view
.
$
(
'
.configure-chatroom-button
'
).
is
(
'
:visible
'
)).
toBeTruthy
();
expect
(
view
.
$
(
'
.configure-chatroom-button
'
).
is
(
'
:visible
'
)).
toBeTruthy
();
expect
(
view
.
$
(
'
.toggle-chatbox-button
'
).
is
(
'
:visible
'
)).
toBeTruthy
();
expect
(
view
.
$
(
'
.toggle-chatbox-button
'
).
is
(
'
:visible
'
)).
toBeTruthy
();
expect
(
view
.
$
(
'
.toggle-bookmark
'
).
is
(
'
:visible
'
)).
toBeTruthy
();
expect
(
view
.
$
(
'
.toggle-bookmark
'
).
is
(
'
:visible
'
)).
toBeTruthy
();
...
@@ -303,7 +323,7 @@
...
@@ -303,7 +323,7 @@
/* Server responds with the configuration form.
/* Server responds with the configuration form.
* See: // http://xmpp.org/extensions/xep-0045.html#example-165
* See: // http://xmpp.org/extensions/xep-0045.html#example-165
*/
*/
var
config_stanza
=
$iq
({
from
:
'
co
n
ven@chat.shakespeare.lit
'
,
var
config_stanza
=
$iq
({
from
:
'
coven@chat.shakespeare.lit
'
,
'
id
'
:
IQ_id
,
'
id
'
:
IQ_id
,
'
to
'
:
'
dummy@localhost/desktop
'
,
'
to
'
:
'
dummy@localhost/desktop
'
,
'
type
'
:
'
result
'
})
'
type
'
:
'
result
'
})
...
@@ -535,6 +555,17 @@
...
@@ -535,6 +555,17 @@
});
});
test_utils
.
openChatRoom
(
converse
,
'
lounge
'
,
'
localhost
'
,
'
dummy
'
);
test_utils
.
openChatRoom
(
converse
,
'
lounge
'
,
'
localhost
'
,
'
dummy
'
);
// We pretend this is a new room, so no disco info is returned.
var
features_stanza
=
$iq
({
from
:
'
lounge@localhost
'
,
'
id
'
:
IQ_id
,
'
to
'
:
'
dummy@localhost/desktop
'
,
'
type
'
:
'
error
'
}).
c
(
'
error
'
,
{
'
type
'
:
'
cancel
'
})
.
c
(
'
item-not-found
'
,
{
'
xmlns
'
:
"
urn:ietf:params:xml:ns:xmpp-stanzas
"
});
converse
.
connection
.
_dataRecv
(
test_utils
.
createRequest
(
features_stanza
));
var
view
=
converse
.
chatboxviews
.
get
(
'
lounge@localhost
'
);
var
view
=
converse
.
chatboxviews
.
get
(
'
lounge@localhost
'
);
spyOn
(
view
,
'
join
'
).
andCallThrough
();
spyOn
(
view
,
'
join
'
).
andCallThrough
();
...
@@ -627,13 +658,15 @@
...
@@ -627,13 +658,15 @@
}));
}));
it
(
"
can be joined automatically, based upon a received invite
"
,
mock
.
initConverse
(
function
(
converse
)
{
it
(
"
can be joined automatically, based upon a received invite
"
,
mock
.
initConverse
(
function
(
converse
)
{
test_utils
.
openChatRoom
(
converse
,
'
lounge
'
,
'
localhost
'
,
'
dummy
'
);
test_utils
.
createContacts
(
converse
,
'
current
'
);
// We need roster contacts, who can invite us
spyOn
(
window
,
'
confirm
'
).
andCallFake
(
function
()
{
spyOn
(
window
,
'
confirm
'
).
andCallFake
(
function
()
{
return
true
;
return
true
;
});
});
test_utils
.
createContacts
(
converse
,
'
current
'
);
// We need roster contacts, who can invite us
test_utils
.
openAndEnterChatRoom
(
converse
,
'
lounge
'
,
'
localhost
'
,
'
dummy
'
);
var
view
=
converse
.
chatboxviews
.
get
(
'
lounge@localhost
'
);
var
view
=
converse
.
chatboxviews
.
get
(
'
lounge@localhost
'
);
view
.
close
();
view
.
close
();
view
.
model
.
destroy
();
// Manually calling this, otherwise we have to mock stanzas.
var
name
=
mock
.
cur_names
[
0
];
var
name
=
mock
.
cur_names
[
0
];
var
from_jid
=
name
.
replace
(
/ /g
,
'
.
'
).
toLowerCase
()
+
'
@localhost
'
;
var
from_jid
=
name
.
replace
(
/ /g
,
'
.
'
).
toLowerCase
()
+
'
@localhost
'
;
var
room_jid
=
'
lounge@localhost
'
;
var
room_jid
=
'
lounge@localhost
'
;
...
@@ -874,23 +907,95 @@
...
@@ -874,23 +907,95 @@
expect
(
$occupants
.
children
().
first
(
0
).
text
()).
toBe
(
"
newnick
"
);
expect
(
$occupants
.
children
().
first
(
0
).
text
()).
toBe
(
"
newnick
"
);
}));
}));
if
(
"
queries for the room information before attempting to join the user
"
,
mock
.
initConverse
(
function
(
converse
)
{
var
sent_IQ
,
IQ_id
;
var
sendIQ
=
converse
.
connection
.
sendIQ
;
spyOn
(
converse
.
connection
,
'
sendIQ
'
).
andCallFake
(
function
(
iq
,
callback
,
errback
)
{
sent_IQ
=
iq
;
IQ_id
=
sendIQ
.
bind
(
this
)(
iq
,
callback
,
errback
);
});
converse_api
.
rooms
.
open
(
'
coven@chat.shakespeare.lit
'
,
{
'
nick
'
:
'
some1
'
});
// Check that the room queried for the feautures.
expect
(
sent_IQ
.
toLocaleString
()).
toBe
(
"
<iq from='dummy@localhost/resource' to='coven@chat.shakespeare.lit' type='get' xmlns='jabber:client' id='
"
+
IQ_id
+
"
'>
"
+
"
<query xmlns='http://jabber.org/protocol/disco#info'/>
"
+
"
</iq>
"
);
/* <iq from='coven@chat.shakespeare.lit'
* id='ik3vs715'
* to='hag66@shakespeare.lit/pda'
* type='result'>
* <query xmlns='http://jabber.org/protocol/disco#info'>
* <identity
* category='conference'
* name='A Dark Cave'
* type='text'/>
* <feature var='http://jabber.org/protocol/muc'/>
* <feature var='muc_passwordprotected'/>
* <feature var='muc_hidden'/>
* <feature var='muc_temporary'/>
* <feature var='muc_open'/>
* <feature var='muc_unmoderated'/>
* <feature var='muc_nonanonymous'/>
* </query>
* </iq>
*/
var
features_stanza
=
$iq
({
from
:
'
coven@chat.shakespeare.lit
'
,
'
id
'
:
IQ_id
,
'
to
'
:
'
dummy@localhost/desktop
'
,
'
type
'
:
'
result
'
})
.
c
(
'
query
'
,
{
'
xmlns
'
:
'
http://jabber.org/protocol/disco#info
'
})
.
c
(
'
identity
'
,
{
'
category
'
:
'
conference
'
,
'
name
'
:
'
A Dark Cave
'
,
'
type
'
:
'
text
'
}).
up
()
.
c
(
'
feature
'
,
{
'
var
'
:
'
http://jabber.org/protocol/muc
'
}).
up
()
.
c
(
'
feature
'
,
{
'
var
'
:
'
passwordprotected
'
}).
up
()
.
c
(
'
feature
'
,
{
'
var
'
:
'
hidden
'
}).
up
()
.
c
(
'
feature
'
,
{
'
var
'
:
'
temporary
'
}).
up
()
.
c
(
'
feature
'
,
{
'
var
'
:
'
open
'
}).
up
()
.
c
(
'
feature
'
,
{
'
var
'
:
'
unmoderated
'
}).
up
()
.
c
(
'
feature
'
,
{
'
var
'
:
'
nonanonymous
'
});
converse
.
connection
.
_dataRecv
(
test_utils
.
createRequest
(
features_stanza
));
var
view
=
converse
.
chatboxviews
.
get
(
'
coven@chat.shakespeare.lit
'
);
expect
(
view
.
model
.
get
(
'
passwordprotected
'
)).
toBe
(
'
true
'
);
expect
(
view
.
model
.
get
(
'
hidden
'
)).
toBe
(
'
true
'
);
expect
(
view
.
model
.
get
(
'
temporary
'
)).
toBe
(
'
true
'
);
expect
(
view
.
model
.
get
(
'
open
'
)).
toBe
(
'
true
'
);
expect
(
view
.
model
.
get
(
'
unmoderated
'
)).
toBe
(
'
true
'
);
expect
(
view
.
model
.
get
(
'
nonanonymous
'
)).
toBe
(
'
true
'
);
}));
it
(
"
indicates when a room is no longer anonymous
"
,
mock
.
initConverse
(
function
(
converse
)
{
it
(
"
indicates when a room is no longer anonymous
"
,
mock
.
initConverse
(
function
(
converse
)
{
converse_api
.
rooms
.
open
(
'
room@conference.example.org
'
,
{
var
sent_IQ
,
IQ_id
;
'
nick
'
:
'
some1
'
,
var
sendIQ
=
converse
.
connection
.
sendIQ
;
'
auto_configure
'
:
true
,
spyOn
(
converse
.
connection
,
'
sendIQ
'
).
andCallFake
(
function
(
iq
,
callback
,
errback
)
{
'
roomconfig
'
:
{
sent_IQ
=
iq
;
'
changesubject
'
:
false
,
IQ_id
=
sendIQ
.
bind
(
this
)(
iq
,
callback
,
errback
);
'
membersonly
'
:
true
,
'
persistentroom
'
:
true
,
'
publicroom
'
:
true
,
'
roomdesc
'
:
'
Welcome to this room
'
,
'
whois
'
:
'
anyone
'
}
});
});
converse_api
.
rooms
.
open
(
'
coven@chat.shakespeare.lit
'
,
{
'
nick
'
:
'
some1
'
});
// We pretend this is a new room, so no disco info is returned.
var
features_stanza
=
$iq
({
from
:
'
coven@chat.shakespeare.lit
'
,
'
id
'
:
IQ_id
,
'
to
'
:
'
dummy@localhost/desktop
'
,
'
type
'
:
'
error
'
}).
c
(
'
error
'
,
{
'
type
'
:
'
cancel
'
})
.
c
(
'
item-not-found
'
,
{
'
xmlns
'
:
"
urn:ietf:params:xml:ns:xmpp-stanzas
"
});
converse
.
connection
.
_dataRecv
(
test_utils
.
createRequest
(
features_stanza
));
var
view
=
converse
.
chatboxviews
.
get
(
'
coven@chat.shakespeare.lit
'
);
/* <message xmlns="jabber:client"
/* <message xmlns="jabber:client"
* type="groupchat"
* type="groupchat"
* to="dummy@localhost/converse.js-27854181"
* to="dummy@localhost/converse.js-27854181"
* from="
room@conference.example.org
">
* from="
coven@chat.shakespeare.lit
">
* <x xmlns="http://jabber.org/protocol/muc#user">
* <x xmlns="http://jabber.org/protocol/muc#user">
* <status code="104"/>
* <status code="104"/>
* <status code="172"/>
* <status code="172"/>
...
@@ -900,12 +1005,11 @@
...
@@ -900,12 +1005,11 @@
var
message
=
$msg
({
var
message
=
$msg
({
type
:
'
groupchat
'
,
type
:
'
groupchat
'
,
to
:
'
dummy@localhost/converse.js-27854181
'
,
to
:
'
dummy@localhost/converse.js-27854181
'
,
from
:
'
room@conference.example.org
'
from
:
'
coven@chat.shakespeare.lit
'
}).
c
(
'
x
'
,
{
xmlns
:
Strophe
.
NS
.
MUC_USER
})
}).
c
(
'
x
'
,
{
xmlns
:
Strophe
.
NS
.
MUC_USER
})
.
c
(
'
status
'
,
{
code
:
'
104
'
}).
up
()
.
c
(
'
status
'
,
{
code
:
'
104
'
}).
up
()
.
c
(
'
status
'
,
{
code
:
'
172
'
});
.
c
(
'
status
'
,
{
code
:
'
172
'
});
converse
.
connection
.
_dataRecv
(
test_utils
.
createRequest
(
message
));
converse
.
connection
.
_dataRecv
(
test_utils
.
createRequest
(
message
));
var
view
=
converse
.
chatboxviews
.
get
(
'
room@conference.example.org
'
);
var
$chat_body
=
view
.
$
(
'
.chatroom-body
'
);
var
$chat_body
=
view
.
$
(
'
.chatroom-body
'
);
expect
(
$chat_body
.
html
().
trim
().
indexOf
(
expect
(
$chat_body
.
html
().
trim
().
indexOf
(
'
<div class="chat-info">This room is now no longer anonymous</div>
'
'
<div class="chat-info">This room is now no longer anonymous</div>
'
...
@@ -1029,7 +1133,11 @@
...
@@ -1029,7 +1133,11 @@
runs
(
function
()
{
runs
(
function
()
{
expect
(
view
.
close
).
toHaveBeenCalled
();
expect
(
view
.
close
).
toHaveBeenCalled
();
expect
(
view
.
leave
).
toHaveBeenCalled
();
expect
(
view
.
leave
).
toHaveBeenCalled
();
expect
(
converse
.
emit
).
toHaveBeenCalledWith
(
'
chatBoxClosed
'
,
jasmine
.
any
(
Object
));
// XXX: After refactoring, the chat box only gets closed
// once we have confirmation from the server. To test this,
// we would have to mock the returned presence stanza.
// See the "leave" method on the ChatRoomView.
// expect(converse.emit).toHaveBeenCalledWith('chatBoxClosed', jasmine.any(Object));
});
});
}));
}));
});
});
...
@@ -1174,18 +1282,17 @@
...
@@ -1174,18 +1282,17 @@
}));
}));
it
(
"
will automatically choose a new nickname if a nickname conflict happens and muc_nickname_from_jid=true
"
,
mock
.
initConverse
(
function
(
converse
)
{
it
(
"
will automatically choose a new nickname if a nickname conflict happens and muc_nickname_from_jid=true
"
,
mock
.
initConverse
(
function
(
converse
)
{
/*
/* <presence
<presence
* from='coven@chat.shakespeare.lit/thirdwitch'
from='coven@chat.shakespeare.lit/thirdwitch'
* id='n13mt3l'
id='n13mt3l'
* to='hag66@shakespeare.lit/pda'
to='hag66@shakespeare.lit/pda'
* type='error'>
type='error'>
* <x xmlns='http://jabber.org/protocol/muc'/>
<x xmlns='http://jabber.org/protocol/muc'/>
* <error by='coven@chat.shakespeare.lit' type='cancel'>
<error by='coven@chat.shakespeare.lit' type='cancel'>
* <conflict xmlns='urn:ietf:params:xml:ns:xmpp-stanzas'/>
<conflict xmlns='urn:ietf:params:xml:ns:xmpp-stanzas'/>
* </error>
</error>
* </presence>
</presence>
*/
*/
submitRoomForm
(
converse
);
submitRoomForm
(
converse
);
converse
.
muc_nickname_from_jid
=
true
;
converse
.
muc_nickname_from_jid
=
true
;
...
...
src/converse-bookmarks.js
View file @
7ad39cfd
...
@@ -104,7 +104,7 @@
...
@@ -104,7 +104,7 @@
if
(
!
_
.
isUndefined
(
model
)
&&
model
.
get
(
'
nick
'
))
{
if
(
!
_
.
isUndefined
(
model
)
&&
model
.
get
(
'
nick
'
))
{
this
.
join
(
this
.
model
.
get
(
'
nick
'
));
this
.
join
(
this
.
model
.
get
(
'
nick
'
));
}
else
{
}
else
{
this
.
__super__
.
checkForReservedNick
.
apply
(
this
,
arguments
);
return
this
.
__super__
.
checkForReservedNick
.
apply
(
this
,
arguments
);
}
}
},
},
...
...
src/converse-chatview.js
View file @
7ad39cfd
...
@@ -662,7 +662,6 @@
...
@@ -662,7 +662,6 @@
// model is going to be destroyed afterwards.
// model is going to be destroyed afterwards.
this
.
model
.
set
(
'
chat_state
'
,
converse
.
INACTIVE
);
this
.
model
.
set
(
'
chat_state
'
,
converse
.
INACTIVE
);
this
.
sendChatState
();
this
.
sendChatState
();
this
.
model
.
destroy
();
this
.
model
.
destroy
();
}
}
this
.
remove
();
this
.
remove
();
...
...
src/converse-muc.js
View file @
7ad39cfd
...
@@ -310,7 +310,21 @@
...
@@ -310,7 +310,21 @@
return
converse
.
chatboxviews
.
showChat
(
return
converse
.
chatboxviews
.
showChat
(
_
.
extend
(
settings
,
{
_
.
extend
(
settings
,
{
'
type
'
:
'
chatroom
'
,
'
type
'
:
'
chatroom
'
,
'
affiliation
'
:
null
'
affiliation
'
:
null
,
'
features_fetched
'
:
false
,
'
hidden
'
:
false
,
'
membersonly
'
:
false
,
'
moderated
'
:
false
,
'
nonanonymous
'
:
false
,
'
open
'
:
false
,
'
passwordprotected
'
:
false
,
'
persistent
'
:
false
,
'
public
'
:
false
,
'
semianonymous
'
:
false
,
'
temporary
'
:
false
,
'
unmoderated
'
:
false
,
'
unsecured
'
:
false
,
'
connection_status
'
:
Strophe
.
Status
.
DISCONNECTED
})
})
);
);
};
};
...
@@ -337,6 +351,7 @@
...
@@ -337,6 +351,7 @@
},
},
initialize
:
function
()
{
initialize
:
function
()
{
var
that
=
this
;
this
.
model
.
messages
.
on
(
'
add
'
,
this
.
onMessageAdded
,
this
);
this
.
model
.
messages
.
on
(
'
add
'
,
this
.
onMessageAdded
,
this
);
this
.
model
.
on
(
'
show
'
,
this
.
show
,
this
);
this
.
model
.
on
(
'
show
'
,
this
.
show
,
this
);
this
.
model
.
on
(
'
destroy
'
,
this
.
hide
,
this
);
this
.
model
.
on
(
'
destroy
'
,
this
.
hide
,
this
);
...
@@ -345,22 +360,18 @@
...
@@ -345,22 +360,18 @@
this
.
model
.
on
(
'
change:name
'
,
this
.
renderHeading
,
this
);
this
.
model
.
on
(
'
change:name
'
,
this
.
renderHeading
,
this
);
this
.
createOccupantsView
();
this
.
createOccupantsView
();
this
.
render
();
this
.
render
().
insertIntoDOM
();
// TODO: hide chat area until messages received.
var
nick
=
this
.
model
.
get
(
'
nick
'
);
// XXX: adding the event below to the declarative events map doesn't work.
if
(
!
nick
)
{
this
.
checkForReservedNick
();
}
else
{
this
.
join
(
nick
);
}
this
.
fetchMessages
().
insertIntoDOM
();
// XXX: adding the event below to the events map above doesn't work.
// The code that gets executed because of that looks like this:
// The code that gets executed because of that looks like this:
// this.$el.on('scroll', '.chat-content', this.markScrolled.bind(this));
// this.$el.on('scroll', '.chat-content', this.markScrolled.bind(this));
// Which for some reason doesn't work.
// Which for some reason doesn't work.
// So working around that fact here:
// So working around that fact here:
this
.
$el
.
find
(
'
.chat-content
'
).
on
(
'
scroll
'
,
this
.
markScrolled
.
bind
(
this
));
this
.
$el
.
find
(
'
.chat-content
'
).
on
(
'
scroll
'
,
this
.
markScrolled
.
bind
(
this
));
converse
.
emit
(
'
chatRoomOpened
'
,
this
);
this
.
getRoomFeatures
().
always
(
function
()
{
that
.
join
().
fetchMessages
();
converse
.
emit
(
'
chatRoomOpened
'
,
that
);
});
},
},
createOccupantsView
:
function
()
{
createOccupantsView
:
function
()
{
...
@@ -447,7 +458,6 @@
...
@@ -447,7 +458,6 @@
* well.
* well.
*/
*/
this
.
leave
();
this
.
leave
();
converse
.
ChatBoxView
.
prototype
.
close
.
apply
(
this
,
arguments
);
},
},
toggleOccupants
:
function
(
ev
,
preserve_state
)
{
toggleOccupants
:
function
(
ev
,
preserve_state
)
{
...
@@ -763,7 +773,7 @@
...
@@ -763,7 +773,7 @@
var
room_now_fully_anon
=
stanza
.
querySelector
(
"
status[code='173']
"
);
var
room_now_fully_anon
=
stanza
.
querySelector
(
"
status[code='173']
"
);
if
(
configuration_changed
||
logging_enabled
||
logging_disabled
||
if
(
configuration_changed
||
logging_enabled
||
logging_disabled
||
room_no_longer_anon
||
room_now_semi_anon
||
room_now_fully_anon
)
{
room_no_longer_anon
||
room_now_semi_anon
||
room_now_fully_anon
)
{
this
.
cache
RoomFeatures
();
this
.
get
RoomFeatures
();
}
}
_
.
compose
(
this
.
onChatRoomMessage
.
bind
(
this
),
this
.
showStatusMessages
.
bind
(
this
))(
stanza
);
_
.
compose
(
this
.
onChatRoomMessage
.
bind
(
this
),
this
.
showStatusMessages
.
bind
(
this
))(
stanza
);
return
true
;
return
true
;
...
@@ -830,6 +840,10 @@
...
@@ -830,6 +840,10 @@
* (String) password: Optional password, if required by
* (String) password: Optional password, if required by
* the room.
* the room.
*/
*/
nick
=
nick
?
nick
:
this
.
model
.
get
(
'
nick
'
);
if
(
!
nick
)
{
return
this
.
checkForReservedNick
();
}
this
.
registerHandlers
();
this
.
registerHandlers
();
if
(
this
.
model
.
get
(
'
connection_status
'
)
===
Strophe
.
Status
.
CONNECTED
)
{
if
(
this
.
model
.
get
(
'
connection_status
'
)
===
Strophe
.
Status
.
CONNECTED
)
{
// We have restored a chat room from session storage,
// We have restored a chat room from session storage,
...
@@ -845,12 +859,14 @@
...
@@ -845,12 +859,14 @@
stanza
.
cnode
(
Strophe
.
xmlElement
(
"
password
"
,
[],
password
));
stanza
.
cnode
(
Strophe
.
xmlElement
(
"
password
"
,
[],
password
));
}
}
this
.
model
.
save
(
'
connection_status
'
,
Strophe
.
Status
.
CONNECTING
);
this
.
model
.
save
(
'
connection_status
'
,
Strophe
.
Status
.
CONNECTING
);
return
converse
.
connection
.
send
(
stanza
);
converse
.
connection
.
send
(
stanza
);
return
this
;
},
},
cleanup
:
function
()
{
cleanup
:
function
()
{
this
.
model
.
save
(
'
connection_status
'
,
Strophe
.
Status
.
DISCONNECTED
);
this
.
model
.
save
(
'
connection_status
'
,
Strophe
.
Status
.
DISCONNECTED
);
this
.
removeHandlers
();
this
.
removeHandlers
();
converse
.
ChatBoxView
.
prototype
.
close
.
apply
(
this
,
arguments
);
},
},
leave
:
function
(
exit_msg
)
{
leave
:
function
(
exit_msg
)
{
...
@@ -863,7 +879,8 @@
...
@@ -863,7 +879,8 @@
this
.
occupantsview
.
model
.
reset
();
this
.
occupantsview
.
model
.
reset
();
this
.
occupantsview
.
model
.
browserStorage
.
_clear
();
this
.
occupantsview
.
model
.
browserStorage
.
_clear
();
if
(
!
converse
.
connection
.
connected
)
{
if
(
!
converse
.
connection
.
connected
||
this
.
model
.
get
(
'
connection_status
'
)
===
Strophe
.
Status
.
DISCONNECTED
)
{
// Don't send out a stanza if we're not connected.
// Don't send out a stanza if we're not connected.
this
.
cleanup
();
this
.
cleanup
();
return
;
return
;
...
@@ -1058,45 +1075,45 @@
...
@@ -1058,45 +1075,45 @@
return
deferred
.
promise
();
return
deferred
.
promise
();
},
},
cache
RoomFeatures
:
function
()
{
get
RoomFeatures
:
function
()
{
/* Fetch the room disco info, parse it and then
/* Fetch the room disco info, parse it and then
* save it on the Backbone.Model of this chat rooms.
* save it on the Backbone.Model of this chat rooms.
*
* See http://xmpp.org/extensions/xep-0045.html#disco-roominfo
*/
*/
var
deferred
=
new
$
.
Deferred
();
var
that
=
this
;
var
that
=
this
;
converse
.
connection
.
disco
.
info
(
this
.
model
.
get
(
'
jid
'
),
null
,
converse
.
connection
.
disco
.
info
(
this
.
model
.
get
(
'
jid
'
),
null
,
function
(
iq
)
{
function
(
iq
)
{
/* <iq from='coven@chat.shakespeare.lit'
/*
* id='ik3vs715'
* See http://xmpp.org/extensions/xep-0045.html#disco-roominfo
* to='hag66@shakespeare.lit/pda'
*
* type='result'>
* <identity
* <query xmlns='http://jabber.org/protocol/disco#info'>
* category='conference'
* <identity
* name='A Dark Cave'
* category='conference'
* type='text'/>
* name='A Dark Cave'
* <feature var='http://jabber.org/protocol/muc'/>
* type='text'/>
* <feature var='muc_passwordprotected'/>
* <feature var='http://jabber.org/protocol/muc'/>
* <feature var='muc_hidden'/>
* <feature var='muc_passwordprotected'/>
* <feature var='muc_temporary'/>
* <feature var='muc_hidden'/>
* <feature var='muc_open'/>
* <feature var='muc_temporary'/>
* <feature var='muc_unmoderated'/>
* <feature var='muc_open'/>
* <feature var='muc_nonanonymous'/>
* <feature var='muc_unmoderated'/>
* <feature var='muc_nonanonymous'/>
* </query>
* </iq>
*/
*/
var
features
=
[];
var
features
=
{
'
features_fetched
'
:
true
};
_
.
each
(
iq
.
querySelectorAll
(
'
feature
'
),
function
(
field
)
{
_
.
each
(
iq
.
querySelectorAll
(
'
feature
'
),
function
(
field
)
{
var
fieldname
=
field
.
getAttribute
(
'
var
'
);
var
fieldname
=
field
.
getAttribute
(
'
var
'
);
if
(
!
fieldname
.
startsWith
(
'
muc_
'
))
{
if
(
!
fieldname
.
startsWith
(
'
muc_
'
))
{
return
;
return
;
}
}
features
.
push
(
fieldname
.
replace
(
'
muc_
'
,
''
))
;
features
[
fieldname
.
replace
(
'
muc_
'
,
''
)]
=
true
;
});
});
that
.
model
.
save
({
'
features
'
:
features
});
that
.
model
.
save
(
features
);
}
return
deferred
.
resolve
();
},
deferred
.
reject
);
);
return
deferred
.
promise
();
},
},
configureChatRoom
:
function
(
ev
)
{
configureChatRoom
:
function
(
ev
)
{
...
@@ -1163,6 +1180,7 @@
...
@@ -1163,6 +1180,7 @@
this
.
onNickNameFound
.
bind
(
this
),
this
.
onNickNameFound
.
bind
(
this
),
this
.
onNickNameNotFound
.
bind
(
this
)
this
.
onNickNameNotFound
.
bind
(
this
)
);
);
return
this
;
},
},
onNickNameFound
:
function
(
iq
)
{
onNickNameFound
:
function
(
iq
)
{
...
@@ -1305,7 +1323,7 @@
...
@@ -1305,7 +1323,7 @@
return
;
return
;
},
},
findAndSaveOwnAffiliation
:
function
(
pres
)
{
saveAffiliationAndRole
:
function
(
pres
)
{
/* Parse the presence stanza for the current user's
/* Parse the presence stanza for the current user's
* affiliation.
* affiliation.
*
*
...
@@ -1320,13 +1338,17 @@
...
@@ -1320,13 +1338,17 @@
// then the Sizzle selector library might still be needed
// then the Sizzle selector library might still be needed
// here.
// here.
var
item
=
$
(
pres
).
find
(
'
x[xmlns="
'
+
Strophe
.
NS
.
MUC_USER
+
'
"] item
'
).
get
(
0
);
var
item
=
$
(
pres
).
find
(
'
x[xmlns="
'
+
Strophe
.
NS
.
MUC_USER
+
'
"] item
'
).
get
(
0
);
if
(
_
.
isUndefined
(
item
))
{
if
(
_
.
isUndefined
(
item
))
{
return
;
}
return
;
}
var
jid
=
item
.
getAttribute
(
'
jid
'
);
var
jid
=
item
.
getAttribute
(
'
jid
'
);
var
affiliation
=
item
.
getAttribute
(
'
affiliation
'
);
if
(
Strophe
.
getBareJidFromJid
(
jid
)
===
converse
.
bare_jid
)
{
if
(
Strophe
.
getBareJidFromJid
(
jid
)
===
converse
.
bare_jid
&&
affiliation
)
{
var
affiliation
=
item
.
getAttribute
(
'
affiliation
'
);
this
.
model
.
save
({
'
affiliation
'
:
affiliation
});
var
role
=
item
.
getAttribute
(
'
role
'
);
if
(
affiliation
)
{
this
.
model
.
save
({
'
affiliation
'
:
affiliation
});
}
if
(
role
)
{
this
.
model
.
save
({
'
role
'
:
role
});
}
}
}
},
},
...
@@ -1480,7 +1502,7 @@
...
@@ -1480,7 +1502,7 @@
*
*
* See http://xmpp.org/extensions/xep-0045.html#createroom-instant
* See http://xmpp.org/extensions/xep-0045.html#createroom-instant
*/
*/
this
.
sendConfiguration
().
then
(
this
.
cache
RoomFeatures
.
bind
(
this
));
this
.
sendConfiguration
().
then
(
this
.
get
RoomFeatures
.
bind
(
this
));
},
},
onChatRoomPresence
:
function
(
pres
)
{
onChatRoomPresence
:
function
(
pres
)
{
...
@@ -1499,7 +1521,7 @@
...
@@ -1499,7 +1521,7 @@
var
new_room
=
pres
.
querySelector
(
"
status[code='201']
"
);
var
new_room
=
pres
.
querySelector
(
"
status[code='201']
"
);
if
(
is_self
)
{
if
(
is_self
)
{
this
.
findAndSaveOwnAffiliation
(
pres
);
this
.
saveAffiliationAndRole
(
pres
);
}
}
if
(
is_self
&&
new_room
)
{
if
(
is_self
&&
new_room
)
{
// This is a new room. It will now be configured
// This is a new room. It will now be configured
...
@@ -1515,17 +1537,22 @@
...
@@ -1515,17 +1537,22 @@
show_status_messages
=
false
;
show_status_messages
=
false
;
}
}
}
}
}
else
if
(
this
.
model
.
get
(
'
connection_status
'
)
!==
Strophe
.
Status
.
CONNECTED
)
{
}
else
if
(
!
this
.
model
.
get
(
'
features_fetched
'
)
&&
// This is not a new room, and this is the first
this
.
model
.
get
(
'
connection_status
'
)
!==
Strophe
.
Status
.
CONNECTED
)
{
// presence received or the room config has
// The features for this room weren't fetched yet, perhaps
// changed.
// because it's a new room without locking (in which
this
.
cacheRoomFeatures
();
// case Prosody doesn't send a 201 status).
// This is the first presence received for the room, so
// a good time to fetch the features.
this
.
getRoomFeatures
();
}
}
if
(
show_status_messages
)
{
if
(
show_status_messages
)
{
this
.
hideSpinner
().
showStatusMessages
(
pres
);
this
.
hideSpinner
().
showStatusMessages
(
pres
);
}
}
this
.
occupantsview
.
updateOccupantsOnPresence
(
pres
);
this
.
occupantsview
.
updateOccupantsOnPresence
(
pres
);
this
.
model
.
save
(
'
connection_status
'
,
Strophe
.
Status
.
CONNECTED
);
if
(
this
.
model
.
get
(
'
role
'
)
!==
'
none
'
)
{
this
.
model
.
save
(
'
connection_status
'
,
Strophe
.
Status
.
CONNECTED
);
}
return
true
;
return
true
;
},
},
...
@@ -2211,7 +2238,7 @@
...
@@ -2211,7 +2238,7 @@
converse
.
chatboxviews
.
each
(
function
(
view
)
{
converse
.
chatboxviews
.
each
(
function
(
view
)
{
if
(
view
.
model
.
get
(
'
type
'
)
===
'
chatroom
'
)
{
if
(
view
.
model
.
get
(
'
type
'
)
===
'
chatroom
'
)
{
view
.
model
.
save
(
'
connection_status
'
,
Strophe
.
Status
.
DISCONNECTED
);
view
.
model
.
save
(
'
connection_status
'
,
Strophe
.
Status
.
DISCONNECTED
);
view
.
join
(
view
.
model
.
get
(
'
nick
'
)
);
view
.
join
();
}
}
});
});
};
};
...
...
tests/utils.js
View file @
7ad39cfd
...
@@ -122,6 +122,16 @@
...
@@ -122,6 +122,16 @@
utils
.
openChatRoom
(
converse
,
room
,
server
);
utils
.
openChatRoom
(
converse
,
room
,
server
);
var
view
=
converse
.
chatboxviews
.
get
(
room
+
'
@
'
+
server
);
var
view
=
converse
.
chatboxviews
.
get
(
room
+
'
@
'
+
server
);
// We pretend this is a new room, so no disco info is returned.
var
features_stanza
=
$iq
({
from
:
'
lounge@localhost
'
,
'
id
'
:
IQ_id
,
'
to
'
:
'
dummy@localhost/desktop
'
,
'
type
'
:
'
error
'
}).
c
(
'
error
'
,
{
'
type
'
:
'
cancel
'
})
.
c
(
'
item-not-found
'
,
{
'
xmlns
'
:
"
urn:ietf:params:xml:ns:xmpp-stanzas
"
});
converse
.
connection
.
_dataRecv
(
utils
.
createRequest
(
features_stanza
));
// The XMPP server returns the reserved nick for this user.
// The XMPP server returns the reserved nick for this user.
var
stanza
=
$iq
({
var
stanza
=
$iq
({
'
type
'
:
'
result
'
,
'
type
'
:
'
result
'
,
...
...
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