123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521 |
- /*
- * Copyright (c) 2006-2015 DMDirc Developers
- *
- * Permission is hereby granted, free of charge, to any person obtaining a copy
- * of this software and associated documentation files (the "Software"), to deal
- * in the Software without restriction, including without limitation the rights
- * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- * copies of the Software, and to permit persons to whom the Software is
- * furnished to do so, subject to the following conditions:
- *
- * The above copyright notice and this permission notice shall be included in
- * all copies or substantial portions of the Software.
- *
- * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
- * SOFTWARE.
- */
-
- package com.dmdirc;
-
- import com.dmdirc.commandparser.CommandType;
- import com.dmdirc.config.ConfigBinding;
- import com.dmdirc.events.ChannelClosedEvent;
- import com.dmdirc.events.ChannelSelfActionEvent;
- import com.dmdirc.events.ChannelSelfJoinEvent;
- import com.dmdirc.events.ChannelSelfMessageEvent;
- import com.dmdirc.events.CommandErrorEvent;
- import com.dmdirc.events.DisplayProperty;
- import com.dmdirc.events.NickListClientAddedEvent;
- import com.dmdirc.events.NickListClientRemovedEvent;
- import com.dmdirc.events.NickListClientsChangedEvent;
- import com.dmdirc.events.NickListUpdatedEvent;
- import com.dmdirc.interfaces.Connection;
- import com.dmdirc.interfaces.GroupChat;
- import com.dmdirc.interfaces.GroupChatUser;
- import com.dmdirc.interfaces.User;
- import com.dmdirc.interfaces.WindowModel;
- import com.dmdirc.interfaces.config.ConfigProviderMigrator;
- import com.dmdirc.parser.common.ChannelListModeItem;
- import com.dmdirc.parser.interfaces.ChannelClientInfo;
- import com.dmdirc.parser.interfaces.ChannelInfo;
- import com.dmdirc.parser.interfaces.Parser;
- import com.dmdirc.ui.core.components.WindowComponent;
- import com.dmdirc.ui.input.TabCompleterFactory;
- import com.dmdirc.ui.input.TabCompletionType;
- import com.dmdirc.ui.messages.BackBufferFactory;
- import com.dmdirc.ui.messages.Styliser;
- import com.dmdirc.util.colours.Colour;
- import com.dmdirc.util.colours.ColourUtils;
-
- import com.google.common.collect.EvictingQueue;
-
- import java.util.ArrayList;
- import java.util.Arrays;
- import java.util.Collection;
- import java.util.Collections;
- import java.util.List;
- import java.util.Optional;
- import java.util.Queue;
- import java.util.stream.Collectors;
-
- import javax.annotation.Nullable;
-
- /**
- * The Channel class represents the client's view of the channel. It handles callbacks for channel
- * events from the parser, maintains the corresponding ChannelWindow, and handles user input for the
- * channel.
- */
- public class Channel extends FrameContainer implements GroupChat {
-
- /** The parser's pChannel class. */
- private ChannelInfo channelInfo;
- /** The connection this channel is on. */
- private final Connection connection;
- /** A list of previous topics we've seen. */
- private final Queue<Topic> topics;
- /** Our event handler. */
- private final ChannelEventHandler eventHandler;
- /** The migrator to use to migrate our config provider. */
- private final ConfigProviderMigrator configMigrator;
- /** Manager used to retrieve {@link GroupChatUser}s */
- private final GroupChatUserManager groupChatUserManager;
- /** Whether we're in this channel or not. */
- private boolean isOnChannel;
- /** Whether we should show mode prefixes in text. */
- @ConfigBinding(domain = "channel", key = "showmodeprefix")
- private volatile boolean showModePrefix;
- /** Whether we should show colours in nicks. */
- @ConfigBinding(domain = "ui", key = "shownickcoloursintext")
- private volatile boolean showColours;
-
- /**
- * Creates a new instance of Channel.
- *
- * @param connection The connection object that this channel belongs to
- * @param newChannelInfo The parser's channel object that corresponds to this channel
- * @param configMigrator The config migrator which provides the config for this channel.
- * @param tabCompleterFactory The factory to use to create tab completers.
- */
- public Channel(
- final Connection connection,
- final ChannelInfo newChannelInfo,
- final ConfigProviderMigrator configMigrator,
- final TabCompleterFactory tabCompleterFactory,
- final BackBufferFactory backBufferFactory,
- final GroupChatUserManager groupChatUserManager) {
- super(connection.getWindowModel(), "channel-inactive",
- newChannelInfo.getName(),
- Styliser.stipControlCodes(newChannelInfo.getName()),
- configMigrator.getConfigProvider(),
- backBufferFactory,
- tabCompleterFactory.getTabCompleter(connection.getWindowModel().getTabCompleter(),
- configMigrator.getConfigProvider(), CommandType.TYPE_CHANNEL,
- CommandType.TYPE_CHAT),
- connection.getWindowModel().getEventBus(),
- Arrays.asList(WindowComponent.TEXTAREA.getIdentifier(),
- WindowComponent.INPUTFIELD.getIdentifier(),
- WindowComponent.TOPICBAR.getIdentifier(),
- WindowComponent.USERLIST.getIdentifier()));
-
- this.configMigrator = configMigrator;
- this.channelInfo = newChannelInfo;
- this.connection = connection;
- this.groupChatUserManager = groupChatUserManager;
-
- getConfigManager().getBinder().bind(this, Channel.class);
-
- topics = EvictingQueue.create(
- getConfigManager().getOptionInt("channel", "topichistorysize"));
-
- eventHandler = new ChannelEventHandler(this, getEventBus(), groupChatUserManager);
-
- initBackBuffer();
- registerCallbacks();
- updateTitle();
- selfJoin();
- }
-
- public ChannelInfo getChannelInfo() {
- return channelInfo;
- }
-
- @Override
- public boolean isOnChannel() {
- return isOnChannel;
- }
-
- /**
- * Registers callbacks with the parser for this channel.
- */
- private void registerCallbacks() {
- eventHandler.registerCallbacks();
- configMigrator.migrate(connection.getProtocol(), connection.getIrcd(), connection.getNetwork(),
- connection.getAddress(), channelInfo.getName());
- }
-
- @Override
- public void sendLine(final String line) {
- if (connection.getState() != ServerState.CONNECTED
- || connection.getParser().get().getChannel(channelInfo.getName()) == null) {
- // We're not in the channel/connected to the server
- return;
- }
-
- final GroupChatUser me = getUser(connection.getLocalUser().get()).get();
- splitLine(line).stream().filter(part -> !part.isEmpty()).forEach(part -> {
- getEventBus().publishAsync(new ChannelSelfMessageEvent(this, me, part));
- channelInfo.sendMessage(part);
- });
- }
-
- @Override
- public int getMaxLineLength() {
- return connection.getState() == ServerState.CONNECTED
- ? connection.getParser().get().getMaxLength("PRIVMSG", getChannelInfo().getName())
- : -1;
- }
-
- @Override
- public void sendAction(final String action) {
- if (connection.getState() != ServerState.CONNECTED
- || connection.getParser().get().getChannel(channelInfo.getName()) == null) {
- // We're not on the server/channel
- return;
- }
-
- if (connection.getParser().get().getMaxLength("PRIVMSG", getChannelInfo().getName())
- <= action.length()) {
- getEventBus().publishAsync(new CommandErrorEvent(this,
- "Warning: action too long to be sent"));
- } else {
- final GroupChatUser me = getUser(connection.getLocalUser().get()).get();
- getEventBus().publishAsync(new ChannelSelfActionEvent(this, me, action));
- channelInfo.sendAction(action);
- }
- }
-
- /**
- * Sets this object's ChannelInfo reference to the one supplied. This only needs to be done if
- * the channel window (and hence this channel object) has stayed open while the user has been
- * out of the channel.
- *
- * @param newChannelInfo The new ChannelInfo object
- */
- public void setChannelInfo(final ChannelInfo newChannelInfo) {
- channelInfo = newChannelInfo;
- registerCallbacks();
- }
-
- /**
- * Called when we join this channel. Just needs to output a message.
- */
- public void selfJoin() {
- isOnChannel = true;
-
- final User me = connection.getLocalUser().get();
- getEventBus().publishAsync(new ChannelSelfJoinEvent(this, me));
-
- setIcon("channel");
-
- connection.getInviteManager().removeInvites(channelInfo.getName());
- }
-
- /**
- * Updates the title of the channel window, and of the main window if appropriate.
- */
- private void updateTitle() {
- String temp = Styliser.stipControlCodes(channelInfo.getName());
-
- if (!channelInfo.getTopic().isEmpty()) {
- temp += " - " + Styliser.stipControlCodes(channelInfo.getTopic());
- }
-
- setTitle(temp);
- }
-
- @Override
- public void join() {
- connection.getParser().get().joinChannel(channelInfo.getName());
- }
-
- @Override
- public void part(final String reason) {
- channelInfo.part(reason);
-
- resetWindow();
- }
-
- @Override
- public void retrieveListModes() {
- channelInfo.requestListModes();
- }
-
- /**
- * Resets the window state after the client has left a channel.
- */
- public void resetWindow() {
- isOnChannel = false;
-
- setIcon("channel-inactive");
-
- // Needs to be published synchronously so that nicklists are cleared before the parser
- // is disconnected (which happens synchronously after this method returns).
- getEventBus().publish(
- new NickListClientsChangedEvent(this, Collections.<GroupChatUser>emptyList()));
- }
-
- @Override
- public void close() {
- super.close();
-
- // Remove any callbacks or listeners
- eventHandler.unregisterCallbacks();
- getConfigManager().getBinder().unbind(this);
-
- connection.getParser().map(Parser::getCallbackManager)
- .ifPresent(cm -> cm.unsubscribe(eventHandler));
-
- // Trigger any actions neccessary
- if (isOnChannel && connection.getState() != ServerState.CLOSING) {
- part(getConfigManager().getOption("general", "partmessage"));
- }
-
- // Trigger action for the window closing
- getEventBus().publish(new ChannelClosedEvent(this));
- }
-
- /**
- * Adds a ChannelClient to this Channel.
- *
- * @param client The client to be added
- */
- public void addClient(final GroupChatUser client) {
- getEventBus().publishAsync(new NickListClientAddedEvent(this, client));
-
- getTabCompleter().addEntry(TabCompletionType.CHANNEL_NICK, client.getNickname());
- }
-
- /**
- * Removes the specified ChannelClient from this channel.
- *
- * @param client The client to be removed
- */
- public void removeClient(final GroupChatUser client) {
- getEventBus().publishAsync(new NickListClientRemovedEvent(this, client));
-
- getTabCompleter().removeEntry(TabCompletionType.CHANNEL_NICK, client.getNickname());
-
- if (client.getUser().equals(connection.getLocalUser().orElse(null))) {
- resetWindow();
- }
- }
-
- /**
- * Replaces the list of known clients on this channel with the specified one.
- *
- * @param clients The list of clients to use
- */
- public void setClients(final Collection<GroupChatUser> clients) {
- getEventBus().publishAsync(new NickListClientsChangedEvent(this, clients));
-
- getTabCompleter().clear(TabCompletionType.CHANNEL_NICK);
-
- getTabCompleter().addEntries(TabCompletionType.CHANNEL_NICK,
- clients.stream().map(GroupChatUser::getNickname).collect(Collectors.toList()));
- }
-
- /**
- * Renames a client that is in this channel.
- *
- * @param oldName The old nickname of the client
- * @param newName The new nickname of the client
- */
- public void renameClient(final String oldName, final String newName) {
- getTabCompleter().removeEntry(TabCompletionType.CHANNEL_NICK, oldName);
- getTabCompleter().addEntry(TabCompletionType.CHANNEL_NICK, newName);
- refreshClients();
- }
-
- @Override
- public void refreshClients() {
- if (!isOnChannel) {
- return;
- }
-
- getEventBus().publishAsync(new NickListUpdatedEvent(this));
- }
-
- /**
- * Returns a string containing the most important mode for the specified client.
- *
- * @param user The channel client to check.
- *
- * @return A string containing the most important mode, or an empty string if there are no
- * (known) modes.
- */
- private String getModes(final GroupChatUser user) {
- if (user == null || !showModePrefix) {
- return "";
- } else {
- return user.getImportantMode();
- }
- }
-
- /**
- * Returns a string[] containing the nickname/ident/host of a channel client.
- *
- *
- *
- * @param client The channel client to check
- *
- * @return A string[] containing displayable components
- * 0 - mode
- * 1 - nickname
- * 2 - ident
- * 3 - hostname
- */
- private String[] getDetails(final GroupChatUser client) {
- if (client == null) {
- // WTF?
- throw new UnsupportedOperationException("getDetails called with"
- + " null ChannelClientInfo");
- }
-
- final String[] res = {
- getModes(client),
- Styliser.CODE_NICKNAME + client.getNickname() + Styliser.CODE_NICKNAME,
- client.getUsername().orElse(""),
- client.getHostname().orElse(""),};
-
- if (showColours) {
- final Optional<Colour> foreground
- = client.getDisplayProperty(DisplayProperty.FOREGROUND_COLOUR);
- final Optional<Colour> background
- = client.getDisplayProperty(DisplayProperty.BACKGROUND_COLOUR);
- if (foreground.isPresent()) {
- String prefix = Styliser.CODE_HEXCOLOUR + ColourUtils.getHex(foreground.get());
- if (background.isPresent()) {
- prefix += ',' + ColourUtils.getHex(background.get());
- }
- res[1] = prefix + res[1] + Styliser.CODE_HEXCOLOUR;
- }
- }
-
- return res;
- }
-
- // ---------------------------------------------------- TOPIC HANDLING -----
- /**
- * Adds the specified topic to this channel's topic list.
- *
- * @param topic The topic to be added.
- */
- public void addTopic(final Topic topic) {
- synchronized (topics) {
- topics.add(topic);
- }
- updateTitle();
- }
-
- @Override
- public List<Topic> getTopics() {
- synchronized (topics) {
- return new ArrayList<>(topics);
- }
- }
-
- @Override
- public Optional<Topic> getCurrentTopic() {
- synchronized (topics) {
- if (topics.isEmpty()) {
- return Optional.empty();
- } else {
- return Optional.of(getTopics().get(topics.size() - 1));
- }
- }
- }
-
- // ------------------------------------------ PARSER METHOD DELEGATION -----
- @Override
- public void setTopic(final String topic) {
- channelInfo.setTopic(topic);
- }
-
- @Override
- public int getMaxTopicLength() {
- return connection.getParser().get().getMaxTopicLength();
- }
-
- @Override
- public Optional<Connection> getConnection() {
- return Optional.of(connection);
- }
-
- @Override
- public Optional<GroupChatUser> getUser(final User user) {
- final ChannelClientInfo ci = channelInfo.getChannelClient(((Client) user).getClientInfo());
- if (ci == null) {
- return Optional.empty();
- }
- return Optional.of(groupChatUserManager.getUserFromClient(ci, user, this));
- }
-
- @Override
- public Collection<GroupChatUser> getUsers() {
- return channelInfo.getChannelClients().stream()
- .map(client -> groupChatUserManager.getUserFromClient(client, this))
- .collect(Collectors.toList());
- }
-
- @Override
- public WindowModel getWindowModel() {
- return this;
- }
-
- @Override
- public void kick(final GroupChatUser user, final Optional<String> reason) {
- ((ChannelClient) user).getClientInfo().kick(
- reason.orElse(getConfigManager().getOption("general", "kickmessage")));
- }
-
- @Override
- public Collection<ChannelListModeItem> getListModeItems(final char mode) {
- return channelInfo.getListMode(mode);
- }
-
- @Override
- public void setMode(final char mode, @Nullable final String value) {
- channelInfo.alterMode(true, mode, value);
- }
-
- @Override
- public void removeMode(final char mode, final String value) {
- channelInfo.alterMode(false, mode, value);
- }
-
- @Override
- public void flushModes() {
- channelInfo.flushModes();
- }
-
- @Override
- public String getModes() {
- return channelInfo.getModes();
- }
-
- @Override
- public String getModeValue(final char mode) {
- return channelInfo.getMode(mode);
- }
-
- @Override
- public void requestUsersInfo() {
- channelInfo.sendWho();
- }
-
- }
|