/* * The MIT License (MIT) * * Copyright (c) 2012-2014 lacolaco.net * * 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 net.lacolaco.smileessence.view.dialog import android.app.AlertDialog import android.app.Dialog import android.os.Bundle import android.text.Html import android.text.TextUtils import android.text.method.LinkMovementMethod import android.view.View import android.widget.AbsListView import android.widget.ListView import com.handmark.pulltorefresh.library.PullToRefreshBase import kotlinx.android.synthetic.main.dialog_user_detail.view.* import net.lacolaco.smileessence.R import net.lacolaco.smileessence.data.ImageCache import net.lacolaco.smileessence.entity.User import net.lacolaco.smileessence.twitter.task.Timelines import net.lacolaco.smileessence.twitter.task.Users import net.lacolaco.smileessence.util.UIHandler import net.lacolaco.smileessence.view.DialogHelper import net.lacolaco.smileessence.view.ThreeStateButton import net.lacolaco.smileessence.view.adapter.TimelineAdapter import org.jetbrains.anko.browse class UserDetailDialogFragment : StackableDialogFragment(), View.OnClickListener, PullToRefreshBase.OnRefreshListener2 { private val adapter by lazy { TimelineAdapter(activity, world) } private lateinit var user: User override fun onClick(v: View) { when (v.id) { R.id.imageview_user_detail_menu -> { val v = view val builder = AlertDialog.Builder(activity) builder.setTitle("@" + user.screenName) .setItems(R.array.user_commands) { dialog, which -> // XXX UIHandler().postDelayed({ if (this@UserDetailDialogFragment.isAdded) { updateRelationship(v) } }, 1000) when (which) { 0 -> { val text = String.format("@%s ", user.screenName) world.postState.beginTransaction().insertText(0, text).moveCursor(text.length).commit() world.notify(R.string.notice_add_to_reply) } 1 -> DialogHelper.showDialog(activity, SendMessageDialogFragment.newInstance(user)) 2 -> ConfirmDialogFragment.show(activity, getString(R.string.dialog_confirm_commands)) { Users.BlockTask(world.account, user.id) .onDone { user -> world.notify(R.string.notice_block_succeeded) } .onFail { ex -> world.notifyError(R.string.notice_block_failed) } .execute() } 3 -> ConfirmDialogFragment.show(activity, getString(R.string.dialog_confirm_commands)) { Users.UnblockTask(world.account, user.id) .onDone { user -> world.notify(R.string.notice_unblock_succeeded) } .onFail { x -> world.notifyError(R.string.notice_unblock_failed) } .execute() } 4 -> ConfirmDialogFragment.show(activity, getString(R.string.dialog_confirm_commands)) { Users.ReportForSpamTask(world.account, user.id) .onDone { user -> world.notify(R.string.notice_r4s_succeeded) } .onFail { ex -> world.notifyError(R.string.notice_r4s_failed) } .execute() } 5 -> browse(user.aclogTimelineURL) else -> throw IllegalStateException() } } val dialog = builder.create() dialog.show() } R.id.imageview_user_detail_icon -> { browse(user.profileImageUrl!!) } R.id.textview_user_detail_screenname, R.id.textview_user_detail_tweet_count -> { browse(user.userHomeURL) } R.id.textview_user_detail_friend_count -> { browse(String.format("%s/following", user.userHomeURL)) } R.id.textview_user_detail_follower_count -> { browse(String.format("%s/followers", user.userHomeURL)) } R.id.textview_user_detail_favorite_count -> { browse(String.format("%s/favorites", user.userHomeURL)) } R.id.button_user_detail_follow -> { val v = view ConfirmDialogFragment.show(activity, getString(R.string.dialog_confirm_commands)) { val account = world.account val isFollowing = v.button_user_detail_follow.state == ThreeStateButton.STATE_ON v.button_user_detail_follow.state = ThreeStateButton.STATE_LOCKED if (isFollowing) { Users.UnfollowTask(account, user.id) .onDoneUI { result -> world.notify(R.string.notice_unfollow_succeeded) updateRelationship(v) } .onFail { x -> world.notifyError(R.string.notice_unfollow_failed) } .execute() } else { Users.FollowTask(account, user.id) .onDoneUI { result -> world.notify(R.string.notice_follow_succeeded) updateRelationship(v) } .onFail { x -> world.notifyError(R.string.notice_follow_failed) } .execute() } } } } } // --------------------- Interface OnRefreshListener2 --------------------- override fun onPullDownToRefresh(refreshView: PullToRefreshBase) { val currentAccount = world.account Timelines.UserTimelineTask(currentAccount, user.id) .setCount(200) .setSinceId(adapter.topID) .onFail { x -> world.notifyError(R.string.notice_error_get_user_timeline) } .onDoneUI { tweets -> adapter.addAll(tweets) updateListView(refreshView.refreshableView, adapter, true) refreshView.onRefreshComplete() } .execute() } override fun onPullUpToRefresh(refreshView: PullToRefreshBase) { val currentAccount = world.account Timelines.UserTimelineTask(currentAccount, user.id) .setCount(200) .setMaxId(adapter.lastID - 1) .onFail { x -> world.notifyError(R.string.notice_error_get_user_timeline) } .onDoneUI { tweets -> adapter.addAll(tweets) updateListView(refreshView.refreshableView, adapter, false) refreshView.onRefreshComplete() } .execute() } override fun onCreateDialog(savedInstanceState: Bundle?): Dialog { val found = User.fetch(arguments.getLong(KEY_USER_ID)) if (found == null) { world.notify(R.string.notice_error_show_user) return DisposeDialog(activity) } user = found val v = activity.layoutInflater.inflate(R.layout.dialog_user_detail, null) v.imageview_user_detail_menu.setOnClickListener(this) v.textview_user_detail_screenname.setOnClickListener(this) v.textview_user_detail_description.movementMethod = LinkMovementMethod.getInstance() v.textview_user_detail_tweet_count.setOnClickListener(this) v.textview_user_detail_friend_count.setOnClickListener(this) v.textview_user_detail_follower_count.setOnClickListener(this) v.textview_user_detail_favorite_count.setOnClickListener(this) v.imageview_user_detail_icon.setOnClickListener(this) v.button_user_detail_follow.setOnClickListener(this) v.listview_user_detail_timeline.setOnRefreshListener(this) v.tabhost.setup() val tab1 = v.tabhost.newTabSpec("tab1").setContent(R.id.tab1).setIndicator(getString(R.string.user_detail_tab_info)) v.tabhost.addTab(tab1) val tab2 = v.tabhost.newTabSpec("tab2").setContent(R.id.tab2).setIndicator(getString(R.string.user_detail_tab_timeline)) v.tabhost.addTab(tab2) v.tabhost.currentTab = 0 updateUserDataBasic(v) updateUserDataDetail(v) v.listview_user_detail_timeline.setAdapter(adapter) executeUserTimelineTask(v, adapter) updateRelationship(v) return AlertDialog.Builder(activity) .setView(v) .setCancelable(true) .create() } private fun executeUserTimelineTask(v: View, adapter: TimelineAdapter) { val account = world.account v.tabhost.tabWidget.getChildTabViewAt(1).visibility = View.GONE Timelines.UserTimelineTask(account, user.id) .setCount(200) .onFail { x -> world.notifyError(R.string.notice_error_get_user_timeline) } .onDoneUI { tweets -> adapter.addAll(tweets) adapter.updateForce() v.tabhost.tabWidget.getChildTabViewAt(1).visibility = View.VISIBLE } .execute() } private val htmlDescription: String? get() { var html = user.description html = html.replace("https?://[\\w/:%#$&?()~.=+-]+".toRegex(), "$0") html = html.replace("@([a-zA-Z0-9_]+)".toRegex(), "$0") html = html.replace("\r\n".toRegex(), "
") return html } private fun updateUserDataBasic(v: View) { v.textview_user_detail_name.text = user.name v.textview_user_detail_screenname.text = user.screenName v.texttview_user_detail_protected.visibility = if (user.isProtected) View.VISIBLE else View.GONE v.imageview_user_detail_icon.setImageUrl(user.profileImageUrl, ImageCache.getImageLoader()) } private fun updateUserDataDetail(v: View) { if (TextUtils.isEmpty(user.location)) { v.textview_user_detail_locate.visibility = View.GONE } else { v.textview_user_detail_locate.text = user.location v.textview_user_detail_locate.visibility = View.VISIBLE } if (TextUtils.isEmpty(user.url)) { v.textview_user_detail_url.visibility = View.GONE } else { v.textview_user_detail_url.text = user.url v.textview_user_detail_url.visibility = View.VISIBLE } v.textview_user_detail_description.text = Html.fromHtml(htmlDescription) v.textview_user_detail_tweet_count.text = user.statusesCount.toString() v.textview_user_detail_friend_count.text = user.friendsCount.toString() v.textview_user_detail_follower_count.text = user.followersCount.toString() v.textview_user_detail_favorite_count.text = user.favoritesCount.toString() v.imageview_user_detail_header.setImageUrl(user.profileBannerUrl, ImageCache.getImageLoader()) } private fun updateListView(absListView: AbsListView, adapter: TimelineAdapter, addedToTop: Boolean) { val before = adapter.count adapter.notifyDataSetChanged() // synchronized call (not adapter#updateForce()) val after = adapter.count val increments = after - before if (increments > 0) { if (addedToTop) { absListView.setSelection(increments + 1) absListView.smoothScrollToPositionFromTop(increments, 0) absListView.setSelection(increments) } else { absListView.smoothScrollToPositionFromTop(before, 0) } } } private fun updateRelationship(v: View) { val account = world.account if (user === account.user) { v.textview_user_detail_followed.setText(R.string.user_detail_followed_is_me) v.button_user_detail_follow.visibility = View.GONE } else { v.button_user_detail_follow.state = ThreeStateButton.STATE_LOCKED v.textview_user_detail_followed.setText(R.string.user_detail_loading) Users.ShowFriendshipTask(account, user.id).onDoneUI { relationship -> val isFollowing = relationship.isSourceFollowingTarget val isFollowed = relationship.isSourceFollowedByTarget v.button_user_detail_follow.state = if (isFollowing) ThreeStateButton.STATE_ON else ThreeStateButton.STATE_OFF v.textview_user_detail_followed.setText(if (isFollowed) R.string.user_detail_followed else R.string.user_detail_not_followed) }.execute() } } companion object { private val KEY_USER_ID = "userID" fun newInstance(user: User): UserDetailDialogFragment { val obj = UserDetailDialogFragment() val bundle = Bundle() bundle.putLong(KEY_USER_ID, user.id) obj.arguments = bundle return obj } } }