package jp.juggler.subwaytooter import android.Manifest import android.annotation.SuppressLint import android.app.Activity import android.app.Dialog import android.content.ContentResolver import android.content.ContentValues import android.content.Intent import android.content.SharedPreferences import android.content.pm.PackageManager import android.graphics.Bitmap import android.net.Uri import android.os.AsyncTask import android.os.Build import android.os.Bundle import android.os.Handler import android.provider.MediaStore import android.support.v13.view.inputmethod.InputConnectionCompat import android.support.v13.view.inputmethod.InputContentInfoCompat import android.support.v4.app.ActivityCompat import android.support.v4.content.ContextCompat import android.support.v7.app.AlertDialog import android.support.v7.app.AppCompatActivity import android.text.Editable import android.text.Spannable import android.text.TextWatcher import android.text.method.LinkMovementMethod import android.view.View import android.view.ViewGroup import android.view.ViewTreeObserver import android.widget.* import jp.juggler.subwaytooter.Styler.defaultColorIcon import jp.juggler.subwaytooter.api.* import jp.juggler.subwaytooter.api.entity.* import jp.juggler.subwaytooter.dialog.* import jp.juggler.subwaytooter.span.MyClickableSpan import jp.juggler.subwaytooter.span.MyClickableSpanClickCallback import jp.juggler.subwaytooter.table.AcctColor import jp.juggler.subwaytooter.table.PostDraft import jp.juggler.subwaytooter.table.SavedAccount import jp.juggler.subwaytooter.util.* import jp.juggler.subwaytooter.view.FocusPointView import jp.juggler.subwaytooter.view.MyEditText import jp.juggler.subwaytooter.view.MyNetworkImageView import jp.juggler.util.* import okhttp3.MediaType import okhttp3.MultipartBody import okhttp3.Request import okhttp3.RequestBody import okio.BufferedSink import org.json.JSONArray import org.json.JSONException import org.json.JSONObject import java.io.* import java.lang.ref.WeakReference import java.util.* import java.util.concurrent.ConcurrentLinkedQueue import java.util.concurrent.atomic.AtomicBoolean class ActPost : AppCompatActivity(), View.OnClickListener, PostAttachment.Callback { companion object { internal val log = LogCategory("ActPost") internal const val EXTRA_POSTED_ACCT = "posted_acct" internal const val EXTRA_POSTED_STATUS_ID = "posted_status_id" internal const val EXTRA_POSTED_REPLY_ID = "posted_reply_id" internal const val EXTRA_POSTED_REDRAFT_ID = "posted_redraft_id" internal const val KEY_ACCOUNT_DB_ID = "account_db_id" internal const val KEY_REPLY_STATUS = "reply_status" internal const val KEY_REDRAFT_STATUS = "redraft_status" internal const val KEY_INITIAL_TEXT = "initial_text" internal const val KEY_SENT_INTENT = "sent_intent" internal const val KEY_QUOTED_RENOTE = "quoted_renote" internal const val KEY_SCHEDULED_STATUS = "scheduled_status" internal const val KEY_ATTACHMENT_LIST = "attachment_list" internal const val KEY_VISIBILITY = "visibility" internal const val KEY_IN_REPLY_TO_ID = "in_reply_to_id" internal const val KEY_IN_REPLY_TO_TEXT = "in_reply_to_text" internal const val KEY_IN_REPLY_TO_IMAGE = "in_reply_to_image" internal const val KEY_IN_REPLY_TO_URL = "in_reply_to_url" private const val REQUEST_CODE_ATTACHMENT = 1 private const val REQUEST_CODE_CAMERA = 2 private const val REQUEST_CODE_MUSHROOM = 3 private const val REQUEST_CODE_VIDEO = 4 private const val REQUEST_CODE_ATTACHMENT_OLD = 5 private const val PERMISSION_REQUEST_CODE = 1 internal const val MIME_TYPE_JPEG = "image/jpeg" internal const val MIME_TYPE_PNG = "image/png" internal val list_resize_max = intArrayOf(0, 640, 800, 1024, 1280, 1600, 2048) internal val acceptable_mime_types = HashSet().apply { // add("image/*") // Android標準のギャラリーが image/* を出してくることがあるらしい add("video/*") // Android標準のギャラリーが image/* を出してくることがあるらしい // add("image/jpeg") add("image/png") add("image/gif") add("video/webm") add("video/mp4") add("video/quicktime") } private val imageHeaderList = arrayOf( Pair("image/jpeg", intArrayOf(0xff, 0xd8, 0xff, 0xe0).toByteArray()), Pair( "image/png", intArrayOf(0x89, 0x50, 0x4E, 0x47, 0x0D, 0x0A, 0x1A, 0x0A).toByteArray() ), Pair("image/gif", charArrayOf('G', 'I', 'F').toByteArray()) ) private fun checkImageHeaderList(contentResolver : ContentResolver, uri : Uri) : String? { try { contentResolver.openInputStream(uri)?.use { inStream -> val data = ByteArray(32) val nRead = inStream.read(data, 0, data.size) for(pair in imageHeaderList) { val type = pair.first val header = pair.second if(nRead >= header.size && data.startWith(header)) return type } } } catch(ex : Throwable) { log.e(ex, "checkImageHeaderList failed.") } return null } // private void performCameraVideo(){ // // try{ // Intent takeVideoIntent = new Intent( MediaStore.ACTION_VIDEO_CAPTURE ); // startActivityForResult( takeVideoIntent, REQUEST_CODE_VIDEO ); // }catch( Throwable ex ){ // warning.trace( ex ); // Utils.showToast( this, ex, "opening video app failed." ); // } // } ///////////////////////////////////////////////// const val DRAFT_CONTENT = "content" const val DRAFT_CONTENT_WARNING = "content_warning" internal const val DRAFT_CONTENT_WARNING_CHECK = "content_warning_check" internal const val DRAFT_NSFW_CHECK = "nsfw_check" internal const val DRAFT_VISIBILITY = "visibility" internal const val DRAFT_ACCOUNT_DB_ID = "account_db_id" internal const val DRAFT_ATTACHMENT_LIST = "attachment_list" internal const val DRAFT_REPLY_ID = "reply_id" internal const val DRAFT_REPLY_TEXT = "reply_text" internal const val DRAFT_REPLY_IMAGE = "reply_image" internal const val DRAFT_REPLY_URL = "reply_url" internal const val DRAFT_IS_ENQUETE = "is_enquete" internal const val DRAFT_ENQUETE_ITEMS = "enquete_items" internal const val DRAFT_QUOTED_RENOTE = "quotedRenote" private const val STATE_MUSHROOM_INPUT = "mushroom_input" private const val STATE_MUSHROOM_START = "mushroom_start" private const val STATE_MUSHROOM_END = "mushroom_end" private const val STATE_REDRAFT_STATUS_ID = "redraft_status_id" private const val STATE_URI_CAMERA_IMAGE = "uri_camera_image" private const val STATE_TIME_SCHEDULE = "time_schedule" private const val STATE_SCHEDULED_STATUS = "scheduled_status" fun open( activity : Activity, request_code : Int, account_db_id : Long, // 再編集する投稿。アカウントと同一のタンスであること redraft_status : TootStatus? = null, // 返信対象の投稿。同一タンス上に同期済みであること reply_status : TootStatus? = null, //初期テキスト initial_text : String? = null, // 外部アプリから共有されたインテント sent_intent : Intent? = null, // (Misskey) 返信を引用リノートにする quotedRenote : Boolean = false, //(Mastodon) 予約投稿の編集 scheduledStatus : TootScheduled? = null ) { val intent = Intent(activity, ActPost::class.java) intent.putExtra(KEY_ACCOUNT_DB_ID, account_db_id) if(redraft_status != null) { intent.putExtra(KEY_REDRAFT_STATUS, redraft_status.json.toString()) } if(reply_status != null) { intent.putExtra(KEY_REPLY_STATUS, reply_status.json.toString()) intent.putExtra(KEY_QUOTED_RENOTE, quotedRenote) } if(initial_text != null) { intent.putExtra(KEY_INITIAL_TEXT, initial_text) } if(sent_intent != null) { intent.putExtra(KEY_SENT_INTENT, sent_intent) } if(scheduledStatus != null) { intent.putExtra(KEY_SCHEDULED_STATUS, scheduledStatus.src.toString()) } activity.startActivityForResult(intent, request_code) } internal fun check_exist(url : String?) : Boolean { if(url?.isEmpty() != false) return false try { val request = Request.Builder().url(url).build() val call = App1.ok_http_client.newCall(request) val response = call.execute() if(response.isSuccessful) { return true } log.e(TootApiClient.formatResponse(response, "check_exist failed.")) } catch(ex : Throwable) { log.trace(ex) } return false } } private lateinit var btnAccount : Button private lateinit var btnVisibility : ImageButton private lateinit var btnAttachment : ImageButton private lateinit var btnPost : ImageButton private lateinit var llAttachment : View private lateinit var ivMedia : List internal lateinit var cbNSFW : CheckBox internal lateinit var cbContentWarning : CheckBox internal lateinit var etContentWarning : MyEditText internal lateinit var etContent : MyEditText internal lateinit var cbQuoteRenote : CheckBox internal lateinit var cbEnquete : CheckBox private lateinit var llEnquete : View internal lateinit var list_etChoice : List private lateinit var tvCharCount : TextView internal lateinit var handler : Handler private lateinit var formRoot : View private lateinit var llReply : View private lateinit var tvReplyTo : TextView private lateinit var btnRemoveReply : ImageButton private lateinit var ivReply : MyNetworkImageView private lateinit var scrollView : ScrollView private lateinit var tvSchedule : TextView private lateinit var ibSchedule : ImageButton private lateinit var ibScheduleReset : ImageButton internal lateinit var pref : SharedPreferences internal lateinit var app_state : AppState private lateinit var post_helper : PostHelper internal var attachment_list = ArrayList() private var isPostComplete : Boolean = false internal var density : Float = 0f private lateinit var account_list : ArrayList private var redraft_status_id : EntityId? = null private var timeSchedule = 0L private var scheduledStatus : TootScheduled? = null private val text_watcher : TextWatcher = object : TextWatcher { override fun beforeTextChanged(charSequence : CharSequence, i : Int, i1 : Int, i2 : Int) { } override fun onTextChanged(charSequence : CharSequence, i : Int, i1 : Int, i2 : Int) { } override fun afterTextChanged(editable : Editable) { updateTextCount() } } private val scroll_listener : ViewTreeObserver.OnScrollChangedListener = ViewTreeObserver.OnScrollChangedListener { post_helper.onScrollChanged() } ////////////////////////////////////////////////////////// // Account internal var account : SavedAccount? = null private var uriCameraImage : Uri? = null ////////////////////////////////////////////////////////////////////// // visibility internal var visibility : TootVisibility? = null ///////////////////////////////////////////////// internal var in_reply_to_id : EntityId? = null internal var in_reply_to_text : String? = null internal var in_reply_to_image : String? = null internal var in_reply_to_url : String? = null private var mushroom_input : Int = 0 private var mushroom_start : Int = 0 private var mushroom_end : Int = 0 private val link_click_listener : MyClickableSpanClickCallback = { _, span -> App1.openBrowser(this@ActPost, span.url) } //////////////////////////////////////////////////////////////// override fun onClick(v : View) { when(v.id) { R.id.btnAccount -> performAccountChooser() R.id.btnVisibility -> performVisibility() R.id.btnAttachment -> openAttachment() R.id.ivMedia1 -> performAttachmentClick(0) R.id.ivMedia2 -> performAttachmentClick(1) R.id.ivMedia3 -> performAttachmentClick(2) R.id.ivMedia4 -> performAttachmentClick(3) R.id.btnPost -> performPost() R.id.btnRemoveReply -> removeReply() R.id.btnMore -> performMore() R.id.btnPlugin -> openMushroom() R.id.btnEmojiPicker -> post_helper.openEmojiPickerFromMore() R.id.ibSchedule -> performSchedule() R.id.ibScheduleReset -> resetSchedule() } } override fun onActivityResult(requestCode : Int, resultCode : Int, data : Intent?) { if(requestCode == REQUEST_CODE_ATTACHMENT_OLD && resultCode == Activity.RESULT_OK) { checkAttachments(data?.handleGetContentResult(contentResolver)) } else if(requestCode == REQUEST_CODE_ATTACHMENT && resultCode == Activity.RESULT_OK) { checkAttachments(data?.handleGetContentResult(contentResolver)) } else if(requestCode == REQUEST_CODE_CAMERA) { if(resultCode != Activity.RESULT_OK) { // 失敗したら DBからデータを削除 val uriCameraImage = this.uriCameraImage if(uriCameraImage != null) { contentResolver.delete(uriCameraImage, null, null) this@ActPost.uriCameraImage = null } } else { // 画像のURL val uri = data?.data ?: uriCameraImage if(uri != null) { addAttachment(uri) } else { showToast(this@ActPost, false, "missing image uri") } } } else if(requestCode == REQUEST_CODE_VIDEO && resultCode == Activity.RESULT_OK) { data?.data?.let { addAttachment(it) } } else if(requestCode == REQUEST_CODE_MUSHROOM && resultCode == Activity.RESULT_OK) { data?.getStringExtra("replace_key")?.let { applyMushroomResult(it) } } super.onActivityResult(requestCode, resultCode, data) } override fun onBackPressed() { saveDraft() super.onBackPressed() } override fun onResume() { super.onResume() MyClickableSpan.link_callback = WeakReference(link_click_listener) } override fun onPause() { super.onPause() // 編集中にホーム画面を押したり他アプリに移動する場合は下書きを保存する // やや過剰な気がするが、自アプリに戻ってくるときにランチャーからアイコンタップされると // メイン画面より上にあるアクティビティはすべて消されてしまうので // このタイミングで保存するしかない if(! isPostComplete) { saveDraft() } } override fun onCreate(savedInstanceState : Bundle?) { var sv : String? super.onCreate(savedInstanceState) App1.setActivityTheme(this, true) app_state = App1.getAppState(this) pref = app_state.pref initUI() // Android 9 から、明示的にフォーカスを当てる必要がある if(savedInstanceState == null) { etContent.requestFocus() } if(account_list.isEmpty()) { showToast(this, true, R.string.please_add_account) finish() return } if(savedInstanceState != null) { mushroom_input = savedInstanceState.getInt(STATE_MUSHROOM_INPUT, 0) mushroom_start = savedInstanceState.getInt(STATE_MUSHROOM_START, 0) mushroom_end = savedInstanceState.getInt(STATE_MUSHROOM_END, 0) redraft_status_id = EntityId.from(savedInstanceState, STATE_REDRAFT_STATUS_ID) timeSchedule = savedInstanceState.getLong(STATE_TIME_SCHEDULE, 0L) savedInstanceState.getString(STATE_URI_CAMERA_IMAGE).mayUri()?.let { uriCameraImage = it } val account_db_id = savedInstanceState.getLong(KEY_ACCOUNT_DB_ID, SavedAccount.INVALID_DB_ID) if(account_db_id != SavedAccount.INVALID_DB_ID) { var i = 0 val ie = account_list.size while(i < ie) { val a = account_list[i] if(a.db_id == account_db_id) { selectAccount(a) break } ++ i } } this.visibility = TootVisibility.fromId(savedInstanceState.getInt(KEY_VISIBILITY, - 1)) val a = account if(a != null) { savedInstanceState.getString(STATE_SCHEDULED_STATUS)?.let { scheduledStatus = parseItem(::TootScheduled, TootParser(this@ActPost, a), JSONObject(it), log) } } if(app_state.attachment_list != null) { val list_in_state = app_state.attachment_list if(list_in_state != null) { // static なデータが残ってるならそれを使う this.attachment_list = list_in_state } // コールバックを新しい画面に差し替える for(pa in attachment_list) { pa.callback = this } } else { sv = savedInstanceState.getString(KEY_ATTACHMENT_LIST) if(sv?.isNotEmpty() == true) { // state から復元する app_state.attachment_list = this.attachment_list this.attachment_list.clear() try { sv.toJsonArray().forEach { if(it !is JSONObject) return@forEach try { attachment_list.add(PostAttachment(TootAttachment.decodeJson(it))) } catch(ex : Throwable) { log.trace(ex) } } } catch(ex : Throwable) { log.trace(ex) } } } this.in_reply_to_id = EntityId.from(savedInstanceState, KEY_IN_REPLY_TO_ID) this.in_reply_to_text = savedInstanceState.getString(KEY_IN_REPLY_TO_TEXT) this.in_reply_to_image = savedInstanceState.getString(KEY_IN_REPLY_TO_IMAGE) this.in_reply_to_url = savedInstanceState.getString(KEY_IN_REPLY_TO_URL) } else { app_state.attachment_list = this.attachment_list this.attachment_list.clear() val intent = intent val account_db_id = intent.getLongExtra(KEY_ACCOUNT_DB_ID, SavedAccount.INVALID_DB_ID) if(account_db_id != SavedAccount.INVALID_DB_ID) { var i = 0 val ie = account_list.size while(i < ie) { val a = account_list[i] if(a.db_id == account_db_id) { selectAccount(a) break } ++ i } } val sent_intent = intent.getParcelableExtra(KEY_SENT_INTENT) if(sent_intent != null) { appendContentText(sent_intent) val action = sent_intent.action when(action) { Intent.ACTION_VIEW -> { val uri = sent_intent.data val type = sent_intent.type if(uri != null) addAttachment(uri, type) } Intent.ACTION_SEND -> { val uri = sent_intent.getParcelableExtra(Intent.EXTRA_STREAM) val type = sent_intent.type if(uri != null) addAttachment(uri, type) } Intent.ACTION_SEND_MULTIPLE -> { val list_uri = sent_intent.getParcelableArrayListExtra(Intent.EXTRA_STREAM) if(list_uri != null) { for(uri in list_uri) { if(uri != null) addAttachment(uri) } } } } } appendContentText(intent.getStringExtra(KEY_INITIAL_TEXT)) val account = this.account sv = intent.getStringExtra(KEY_REPLY_STATUS) if(sv != null && account != null) { try { val reply_status = TootParser(this@ActPost, account).status(sv.toJsonObject()) val isQuoterRenote = intent.getBooleanExtra(KEY_QUOTED_RENOTE, false) if(reply_status != null) { if(isQuoterRenote) { cbQuoteRenote.isChecked = true // 引用リノートはCWやメンションを引き継がない } else { // CW をリプライ元に合わせる if(reply_status.spoiler_text.isNotEmpty()) { cbContentWarning.isChecked = true etContentWarning.setText(reply_status.spoiler_text) } val mention_list = ArrayList() val old_mentions = reply_status.mentions if(old_mentions != null) { for(mention in old_mentions) { val who_acct = mention.acct if(who_acct.isNotEmpty()) { if(account.isMe(who_acct)) continue sv = "@" + account.getFullAcct(who_acct) if(! mention_list.contains(sv)) { mention_list.add(sv) } } } } // 元レスのacctを追加する val who_acct = account.getFullAcct(reply_status.account) if(! account.isMe(reply_status.account) // 自己レスにはメンションを追加しない && ! mention_list.contains("@$who_acct") // 既に含まれているならメンションを追加しない ) { mention_list.add("@$who_acct") } val sb = StringBuilder() for(acct in mention_list) { if(sb.isNotEmpty()) sb.append(' ') sb.append(acct) } if(sb.isNotEmpty()) { appendContentText(sb.append(' ').toString()) } } // リプライ表示をつける in_reply_to_id = reply_status.id in_reply_to_text = reply_status.content in_reply_to_image = reply_status.account.avatar_static in_reply_to_url = reply_status.url // 公開範囲 try { // 比較する前にデフォルトの公開範囲を計算する visibility = visibility ?: account.visibility ?: TootVisibility.Public // VISIBILITY_WEB_SETTING だと 1.5未満のタンスでトラブルになる if(TootVisibility.WebSetting == visibility) { // 「Web設定に合わせる」だった場合は無条件にリプライ元の公開範囲に変更する this.visibility = reply_status.visibility } else { // デフォルトの方が公開範囲が大きい場合、リプライ元に合わせて公開範囲を狭める if(TootVisibility.isVisibilitySpoilRequired( this.visibility, reply_status.visibility )) { this.visibility = reply_status.visibility } } } catch(ex : Throwable) { log.trace(ex) } } } catch(ex : Throwable) { log.trace(ex) } } appendContentText(account?.default_text, selectBefore = true) // 再編集 sv = intent.getStringExtra(KEY_REDRAFT_STATUS) if(sv != null && account != null) { try { val base_status = TootParser(this@ActPost, account).status(sv.toJsonObject()) if(base_status != null) { redraft_status_id = base_status.id this.visibility = base_status.visibility val src_attachments = base_status.media_attachments if(src_attachments?.isNotEmpty() == true) { app_state.attachment_list = this.attachment_list this.attachment_list.clear() try { for(src in src_attachments) { if(src is TootAttachment) { src.redraft = true val pa = PostAttachment(src) pa.status = PostAttachment.STATUS_UPLOADED this.attachment_list.add(pa) } } } catch(ex : Throwable) { log.trace(ex) } } // 再編集の場合はdefault_textは反映されない val decodeOptions = DecodeOptions(this) var text : Spannable text = decodeOptions.decodeHTML(base_status.content) etContent.text = text etContent.setSelection(text.length) text = decodeOptions.decodeEmoji(base_status.spoiler_text) etContentWarning.setText(text) etContentWarning.setSelection(text.length) cbContentWarning.isChecked = text.isNotEmpty() cbNSFW.isChecked = base_status.sensitive == true val src_enquete = base_status.enquete val src_items = src_enquete?.items if(src_items != null && src_enquete.type == NicoEnquete.TYPE_ENQUETE) { cbEnquete.isChecked = true text = decodeOptions.decodeHTML(src_enquete.question) etContent.text = text etContent.setSelection(text.length) var src_index = 0 for(et in list_etChoice) { if(src_index < src_items.size) { val choice = src_items[src_index] if(src_index == src_items.size - 1 && choice.text == "\uD83E\uDD14") { // :thinking_face: は再現しない } else { et.setText(decodeOptions.decodeEmoji(choice.text)) ++ src_index continue } } et.setText("") } } } } catch(ex : Throwable) { log.trace(ex) } } // 予約編集の再編集 sv = intent.getStringExtra(KEY_SCHEDULED_STATUS) if(sv != null && account != null) { try { val item = parseItem( ::TootScheduled, TootParser(this@ActPost, account), JSONObject(sv), log ) if(item != null) { scheduledStatus = item timeSchedule = item.timeScheduledAt val text = item.text etContent.setText(text) val cw = item.spoiler_text if(cw?.isNotEmpty() == true) { etContentWarning.setText(cw) cbContentWarning.isChecked = true } else { cbContentWarning.isChecked = false } cbNSFW.isChecked = item.sensitive visibility = item.visibility // 2019/1/7 どうも添付データを古い投稿から引き継げないようだ…。 // バグ臭い // val src_attachments = item.media_attachments // if(src_attachments?.isNotEmpty() == true) { // app_state.attachment_list = this.attachment_list // this.attachment_list.clear() // try { // for(src in src_attachments) { // if(src is TootAttachment) { // src.redraft = true // val pa = PostAttachment(src) // pa.status = PostAttachment.STATUS_UPLOADED // this.attachment_list.add(pa) // } // } // } catch(ex : Throwable) { // log.trace(ex) // } // } } } catch(ex : Throwable) { log.trace(ex) } } } visibility = visibility ?: account?.visibility ?: TootVisibility.Public // 2017/9/13 VISIBILITY_WEB_SETTING から VISIBILITY_PUBLICに変更した // VISIBILITY_WEB_SETTING だと 1.5未満のタンスでトラブルになるので… if(this.account == null) { // 表示を未選択に更新 selectAccount(null) } updateContentWarning() showMediaAttachment() showVisibility() updateTextCount() showReplyTo() showEnquete() showQuotedRenote() showSchedule() } override fun onDestroy() { post_helper.onDestroy() attachment_worker?.cancel() super.onDestroy() } override fun onSaveInstanceState(outState : Bundle?) { super.onSaveInstanceState(outState) outState ?: return outState.putInt(STATE_MUSHROOM_INPUT, mushroom_input) outState.putInt(STATE_MUSHROOM_START, mushroom_start) outState.putInt(STATE_MUSHROOM_END, mushroom_end) redraft_status_id?.putTo(outState, STATE_REDRAFT_STATUS_ID) outState.putLong(STATE_TIME_SCHEDULE, timeSchedule) if(uriCameraImage != null) { outState.putString(STATE_URI_CAMERA_IMAGE, uriCameraImage.toString()) } val account = this.account if(account != null) { outState.putLong(KEY_ACCOUNT_DB_ID, account.db_id) } visibility?.let { outState.putInt(KEY_VISIBILITY, it.id) } if(! attachment_list.isEmpty()) { val array = JSONArray() for(pa in attachment_list) { // アップロード完了したものだけ保持する if(pa.status != PostAttachment.STATUS_UPLOADED) continue val json = pa.attachment?.encodeJson() ?: continue array.put(json) } outState.putString(KEY_ATTACHMENT_LIST, array.toString()) } in_reply_to_id?.putTo(outState, KEY_IN_REPLY_TO_ID) outState.putString(KEY_IN_REPLY_TO_TEXT, in_reply_to_text) outState.putString(KEY_IN_REPLY_TO_IMAGE, in_reply_to_image) outState.putString(KEY_IN_REPLY_TO_URL, in_reply_to_url) } override fun onRestoreInstanceState(savedInstanceState : Bundle) { super.onRestoreInstanceState(savedInstanceState) updateContentWarning() showMediaAttachment() showVisibility() updateTextCount() showReplyTo() showEnquete() showQuotedRenote() } private fun appendContentText( src : String?, selectBefore : Boolean = false ) { if(src?.isEmpty() != false) return val svEmoji = DecodeOptions(context = this, decodeEmoji = true).decodeEmoji(src) if(svEmoji.isEmpty()) return val editable = etContent.text if(editable == null) { val sb = StringBuilder() if(selectBefore) { val start = 0 sb.append(' ') sb.append(svEmoji) etContent.setText(sb) etContent.setSelection(start) } else { sb.append(svEmoji) etContent.setText(sb) etContent.setSelection(sb.length) } } else { if(editable.isNotEmpty() && ! CharacterGroup.isWhitespace(editable[editable.length - 1].toInt()) ) { editable.append(' ') } if(selectBefore) { val start = editable.length editable.append(' ') editable.append(svEmoji) etContent.text = editable etContent.setSelection(start) } else { editable.append(svEmoji) etContent.text = editable etContent.setSelection(editable.length) } } } private fun appendContentText(src : Intent) { val list = ArrayList() var sv : String? sv = src.getStringExtra(Intent.EXTRA_SUBJECT) if(sv?.isNotEmpty() == true) list.add(sv) sv = src.getStringExtra(Intent.EXTRA_TEXT) if(sv?.isNotEmpty() == true) list.add(sv) if(list.isNotEmpty()) { appendContentText(list.joinToString(" ")) } } private fun initUI() { handler = Handler() density = resources.displayMetrics.density setContentView(R.layout.act_post) if(Pref.bpPostButtonBarTop(this)) { val bar = findViewById(R.id.llFooterBar) val parent = bar.parent as ViewGroup parent.removeView(bar) parent.addView(bar, 0) } Styler.fixHorizontalMargin(findViewById(R.id.scrollView)) Styler.fixHorizontalMargin(findViewById(R.id.llFooterBar)) formRoot = findViewById(R.id.viewRoot) scrollView = findViewById(R.id.scrollView) btnAccount = findViewById(R.id.btnAccount) btnVisibility = findViewById(R.id.btnVisibility) btnAttachment = findViewById(R.id.btnAttachment) btnPost = findViewById(R.id.btnPost) llAttachment = findViewById(R.id.llAttachment) cbNSFW = findViewById(R.id.cbNSFW) cbContentWarning = findViewById(R.id.cbContentWarning) etContentWarning = findViewById(R.id.etContentWarning) etContent = findViewById(R.id.etContent) cbQuoteRenote = findViewById(R.id.cbQuoteRenote) cbEnquete = findViewById(R.id.cbEnquete) llEnquete = findViewById(R.id.llEnquete) ivMedia = listOf( findViewById(R.id.ivMedia1), findViewById(R.id.ivMedia2), findViewById(R.id.ivMedia3), findViewById(R.id.ivMedia4) ) list_etChoice = listOf( findViewById(R.id.etChoice1), findViewById(R.id.etChoice2), findViewById(R.id.etChoice3), findViewById(R.id.etChoice4) ) tvCharCount = findViewById(R.id.tvCharCount) llReply = findViewById(R.id.llReply) tvReplyTo = findViewById(R.id.tvReplyTo) btnRemoveReply = findViewById(R.id.btnRemoveReply) ivReply = findViewById(R.id.ivReply) tvSchedule = findViewById(R.id.tvSchedule) ibSchedule = findViewById(R.id.ibSchedule) ibScheduleReset = findViewById(R.id.ibScheduleReset) ibSchedule.setOnClickListener(this) ibScheduleReset.setOnClickListener(this) account_list = SavedAccount.loadAccountList(this@ActPost) SavedAccount.sort(account_list) btnAccount.setOnClickListener(this) btnVisibility.setOnClickListener(this) btnAttachment.setOnClickListener(this) btnPost.setOnClickListener(this) btnRemoveReply.setOnClickListener(this) val btnPlugin : ImageButton = findViewById(R.id.btnPlugin) val btnEmojiPicker : ImageButton = findViewById(R.id.btnEmojiPicker) val btnMore : ImageButton = findViewById(R.id.btnMore) btnPlugin.setOnClickListener(this) btnEmojiPicker.setOnClickListener(this) btnMore.setOnClickListener(this) for(iv in ivMedia) { iv.setOnClickListener(this) iv.setDefaultImage(defaultColorIcon(this, R.drawable.ic_upload)) iv.setErrorImage(defaultColorIcon(this, R.drawable.ic_unknown)) } // setIcon(btnPost, R.drawable.ic_send) // setIcon(btnMore, R.drawable.ic_more) // setIcon(btnPlugin, R.drawable.ic_extension) // setIcon(btnEmojiPicker, R.drawable.ic_face) // setIcon(btnAttachment, R.drawable.ic_attachment) // setIcon(ibSchedule, R.drawable.ic_edit) // setIcon(ibScheduleReset, R.drawable.ic_close) // setIcon(btnRemoveReply, R.drawable.ic_close) cbContentWarning.setOnCheckedChangeListener { _, _ -> updateContentWarning() } cbEnquete.setOnCheckedChangeListener { _, _ -> showEnquete() updateTextCount() } post_helper = PostHelper(this, pref, app_state.handler) post_helper.attachEditText(formRoot, etContent, false, object : PostHelper.Callback2 { override fun onTextUpdate() { updateTextCount() } override fun canOpenPopup() : Boolean { return true } }) etContentWarning.addTextChangedListener(text_watcher) for(et in list_etChoice) { et.addTextChangedListener(text_watcher) } scrollView.viewTreeObserver.addOnScrollChangedListener(scroll_listener) etContent.contentMineTypeArray = acceptable_mime_types.toArray(arrayOfNulls(ActPost.acceptable_mime_types.size)) etContent.commitContentListener = commitContentListener } private var lastInstanceTask : TootTaskRunner? = null private fun getMaxCharCount() : Int { val account = account when { account == null || account.isPseudo -> { } else -> { val info = account.instance // 情報がないか古いなら再取得 if(info == null || System.currentTimeMillis() - info.time_parse >= 300000L) { // 同時に実行するタスクは1つまで var lastTask = lastInstanceTask if(lastTask?.isActive != true) { lastTask = TootTaskRunner(this, TootTaskRunner.PROGRESS_NONE) lastInstanceTask = lastTask lastTask.run(account, object : TootTask { var newInfo : TootInstance? = null override fun background(client : TootApiClient) : TootApiResult? { val result = if(account.isMisskey) { client.request( "/api/meta", account.putMisskeyApiToken().toPostRequestBuilder() ) } else { client.request("/api/v1/instance") } newInfo = TootParser(this@ActPost, account).instance(result?.jsonObject) return result } override fun handleResult(result : TootApiResult?) { if(isFinishing || isDestroyed) return if(newInfo != null) { account.instance = newInfo updateTextCount() } } }) // fall thru } } val max = info?.max_toot_chars if(max != null && max > 0) return max } } return 500 } private fun updateTextCount() { var length = 0 var s = EmojiDecoder.decodeShortCode(etContent.text.toString()) length += s.codePointCount(0, s.length) s = if(cbContentWarning.isChecked) EmojiDecoder.decodeShortCode(etContentWarning.text.toString()) else "" length += s.codePointCount(0, s.length) var max = getMaxCharCount() if(cbEnquete.isChecked) { max -= 150 // フレニコ固有。500-150で350になる for(et in list_etChoice) { s = EmojiDecoder.decodeShortCode(et.text.toString()) length += s.codePointCount(0, s.length) } } val remain = max - length tvCharCount.text = Integer.toString(remain) val color = getAttributeColor( this, if(remain < 0) R.attr.colorRegexFilterError else android.R.attr.textColorPrimary ) tvCharCount.setTextColor(color) } private fun updateContentWarning() { etContentWarning.visibility = if(cbContentWarning.isChecked) View.VISIBLE else View.GONE } internal fun selectAccount(a : SavedAccount?) { this.account = a if(a == null) { post_helper.setInstance(null, false) btnAccount.text = getString(R.string.not_selected) btnAccount.setTextColor(getAttributeColor(this, android.R.attr.textColorPrimary)) btnAccount.setBackgroundResource(R.drawable.btn_bg_transparent) } else { post_helper.setInstance(a.host, a.isMisskey) // 先読みしてキャッシュに保持しておく App1.custom_emoji_lister.getList(a.host, a.isMisskey) { // 何もしない } val acct = a.acct val ac = AcctColor.load(acct) val nickname = if(AcctColor.hasNickname(ac)) ac.nickname else acct btnAccount.text = nickname if(AcctColor.hasColorBackground(ac)) { btnAccount.setBackgroundColor(ac.color_bg) } else { btnAccount.setBackgroundResource(R.drawable.btn_bg_transparent) } if(AcctColor.hasColorForeground(ac)) { btnAccount.setTextColor(ac.color_fg) } else { btnAccount.setTextColor( getAttributeColor( this, android.R.attr.textColorPrimary ) ) } } updateTextCount() } private fun performAccountChooser() { if(scheduledStatus != null) { // 予約投稿の再編集ではアカウントを切り替えられない showToast(this, false, R.string.cant_change_account_when_editing_scheduled_status) return } if(! attachment_list.isEmpty()) { // 添付ファイルがあったら確認の上添付ファイルを捨てないと切り替えられない showToast(this, false, R.string.cant_change_account_when_attachment_specified) return } if(redraft_status_id != null) { // 添付ファイルがあったら確認の上添付ファイルを捨てないと切り替えられない showToast(this, false, R.string.cant_change_account_when_redraft) return } AccountPicker.pick( this, bAllowPseudo = false, bAuto = false, message = getString(R.string.choose_account) ) { ai -> // 別タンスのアカウントに変更したならならin_reply_toの変換が必要 if(in_reply_to_id != null && ! ai.host.equals(account?.host, ignoreCase = true)) { startReplyConversion(ai) } else { setAccountWithVisibilityConversion(ai) } } // final ArrayList< SavedAccount > tmp_account_list = new ArrayList<>(); // tmp_account_list.addAll( account_list ); // // String[] caption_list = new String[ tmp_account_list.size() ]; // for( int i = 0, ie = tmp_account_list.size() ; i < ie ; ++ i ){ // caption_list[ i ] = tmp_account_list.get( i ).acct; // } // // new AlertDialog.Builder( this ) // .setTitle( R.string.choose_account ) // .setItems( caption_list, new DialogInterface.OnClickListener() { // @Override // public void onClick( DialogInterface dialog, int which ){ // // if( which < 0 || which >= tmp_account_list.size() ){ // // 範囲外 // return; // } // // SavedAccount ai = tmp_account_list.get( which ); // // if( ! ai.host.equals( account.host ) ){ // // 別タンスへの移動 // if( in_reply_to_id != - 1L ){ // // 別タンスのアカウントならin_reply_toの変換が必要 // startReplyConversion( ai ); // // } // } // // // リプライがないか、同タンスへの移動 // setAccountWithVisibilityConversion( ai ); // } // } ) // .setNegativeButton( R.string.cancel, null ) // .show(); } internal fun setAccountWithVisibilityConversion(a : SavedAccount) { selectAccount(a) try { if(TootVisibility.isVisibilitySpoilRequired(this.visibility, a.visibility)) { showToast(this@ActPost, true, R.string.spoil_visibility_for_account) this.visibility = a.visibility } } catch(ex : Throwable) { log.trace(ex) } showVisibility() showQuotedRenote() updateTextCount() } @SuppressLint("StaticFieldLeak") private fun startReplyConversion(access_info : SavedAccount) { val in_reply_to_url = this.in_reply_to_url if(in_reply_to_url == null) { // 下書きが古い形式の場合、URLがないので別タンスへの移動ができない AlertDialog.Builder(this@ActPost) .setMessage(R.string.account_change_failed_old_draft_has_no_in_reply_to_url) .setNeutralButton(R.string.close, null) .show() return } TootTaskRunner(this) .progressPrefix(getString(R.string.progress_synchronize_toot)) .run(access_info, object : TootTask { var target_status : TootStatus? = null override fun background(client : TootApiClient) : TootApiResult? { val result = client.syncStatus(access_info, in_reply_to_url) if(result?.data != null) { target_status = result.data as? TootStatus if(target_status == null) { return TootApiResult(getString(R.string.status_id_conversion_failed)) } } return result } override fun handleResult(result : TootApiResult?) { if(result == null) return // cancelled. val target_status = this.target_status if(target_status != null) { in_reply_to_id = target_status.id setAccountWithVisibilityConversion(access_info) } else { showToast( this@ActPost, true, getString(R.string.in_reply_to_id_conversion_failed) + "\n" + result.error ) } } }) } ////////////////////////////////////////////////////////// // Attachment private fun showMediaAttachment() { if(isFinishing) return if(attachment_list.isEmpty()) { llAttachment.visibility = View.GONE } else { llAttachment.visibility = View.VISIBLE ivMedia.forEachIndexed { i, v -> showAttachment_sub(v, i) } } } private fun showAttachment_sub(iv : MyNetworkImageView, idx : Int) { if(idx >= attachment_list.size) { iv.visibility = View.GONE } else { iv.visibility = View.VISIBLE val pa = attachment_list[idx] val a = pa.attachment if(pa.status == PostAttachment.STATUS_UPLOADED && a != null) { iv.setImageUrl(pref, Styler.calcIconRound(iv.layoutParams.width), a.preview_url) } else { iv.setImageUrl(pref, Styler.calcIconRound(iv.layoutParams.width), null) } } } // 添付した画像をタップ private fun performAttachmentClick(idx : Int) { val pa = try { attachment_list[idx] } catch(ex : Throwable) { showToast(this, false, ex.withCaption("can't get attachment item[$idx].")) return } AlertDialog.Builder(this) .setTitle(R.string.media_attachment) .setItems( arrayOf( getString(R.string.set_description), getString(R.string.set_focus_point), getString(R.string.delete) ) ) { _, i -> when(i) { 0 -> editAttachmentDescription(pa) 1 -> openFocusPoint(pa) 2 -> deleteAttachment(pa) } } .setNegativeButton(R.string.cancel, null) .show() } private fun openFocusPoint(pa : PostAttachment) { val attachment = pa.attachment if(attachment != null) { DlgFocusPoint(this, attachment) .setCallback(object : FocusPointView.Callback { override fun onFocusPointUpdate(x : Float, y : Float) { val account = this@ActPost.account ?: return TootTaskRunner(this@ActPost, TootTaskRunner.PROGRESS_NONE).run(account, object : TootTask { override fun background(client : TootApiClient) : TootApiResult? { try { val result = client.request( "/api/v1/media/${attachment.id}", JSONObject() .put("focus", "%.2f,%.2f".format(x, y)) .toPutRequestBuilder() ) new_attachment = parseItem( ::TootAttachment, ServiceType.MASTODON, result?.jsonObject ) return result } catch(ex : Throwable) { return TootApiResult(ex.withCaption("set focus point failed.")) } } var new_attachment : TootAttachment? = null override fun handleResult(result : TootApiResult?) { result ?: return if(new_attachment != null) { pa.attachment = attachment } else { showToast(this@ActPost, true, result.error) } } }) } }) .show() } } private fun deleteAttachment(pa : PostAttachment) { AlertDialog.Builder(this) .setTitle(R.string.confirm_delete_attachment) .setPositiveButton(R.string.ok) { _, _ -> try { attachment_list.remove(pa) } catch(ignored : Throwable) { } showMediaAttachment() } .setNegativeButton(R.string.cancel, null) .show() } private fun editAttachmentDescription(pa : PostAttachment) { val a = pa.attachment if(a == null) { showToast(this, true, R.string.attachment_description_cant_edit_while_uploading) return } DlgTextInput.show( this, getString(R.string.attachment_description), a.description, object : DlgTextInput.Callback { override fun onOK(dialog : Dialog, text : String) { setAttachmentDescription(pa, dialog, text) } override fun onEmptyError() { showToast(this@ActPost, true, R.string.description_empty) } }) } @SuppressLint("StaticFieldLeak") private fun setAttachmentDescription(pa : PostAttachment, dialog : Dialog, text : String) { val attachment_id = pa.attachment?.id ?: return TootTaskRunner(this).run(this@ActPost.account ?: return, object : TootTask { var new_attachment : TootAttachment? = null override fun background(client : TootApiClient) : TootApiResult? { val result = client.request( "/api/v1/media/$attachment_id", JSONObject() .put("description", text) .toPutRequestBuilder() ) new_attachment = parseItem(::TootAttachment, ServiceType.MASTODON, result?.jsonObject) return result } override fun handleResult(result : TootApiResult?) { if(result == null) return // cancelled. val new_attachment = this.new_attachment if(new_attachment != null) { pa.attachment = new_attachment showMediaAttachment() try { dialog.dismiss() } catch(ignored : Throwable) { } } else { showToast(this@ActPost, true, result.error) } } }) } private fun openAttachment() { if(attachment_list.size >= 4) { showToast(this, false, R.string.attachment_too_many) return } if(account == null) { showToast(this, false, R.string.account_select_please) return } val permissionCheck = ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE) if(permissionCheck != PackageManager.PERMISSION_GRANTED) { preparePermission() return } // permissionCheck = ContextCompat.checkSelfPermission( this, Manifest.permission.CAMERA ); // if( permissionCheck != PackageManager.PERMISSION_GRANTED ){ // preparePermission(); // return; // } val a = ActionsDialog() a.addAction(getString(R.string.pick_images)) { performAttachmentOld() } a.addAction(getString(R.string.image_capture)) { performCamera() } // a.addAction( getString( R.string.video_capture ), new Runnable() { // @Override public void run(){ // performCameraVideo(); // } // } ); a.show(this, null) } private fun performAttachmentOld() { // SAFのIntentで開く try { val intent = intentGetContent( true, getString(R.string.pick_images) , "image/*" , "video/*" ) startActivityForResult(intent, REQUEST_CODE_ATTACHMENT_OLD) } catch(ex : Throwable) { log.trace(ex) showToast(this, ex, "ACTION_GET_CONTENT failed.") } } internal interface InputStreamOpener { val mimeType : String @Throws(IOException::class) fun open() : InputStream fun deleteTempFile() } private fun createOpener(uri : Uri, mime_type : String) : InputStreamOpener { while(true) { try { // 画像の種別 val is_jpeg = MIME_TYPE_JPEG == mime_type val is_png = MIME_TYPE_PNG == mime_type if(! is_jpeg && ! is_png) { log.d("createOpener: source is not jpeg or png") break } // 設定からリサイズ指定を読む val resize_to = list_resize_max[Pref.ipResizeImage(pref)] val bitmap = createResizedBitmap( this, uri, resize_to, skipIfNoNeedToResizeAndRotate = true ) if(bitmap != null) { try { val cache_dir = externalCacheDir if(cache_dir == null) { showToast(this, false, "getExternalCacheDir returns null.") break } cache_dir.mkdir() val temp_file = File(cache_dir, "tmp." + Thread.currentThread().id) FileOutputStream(temp_file).use { os -> if(is_jpeg) { bitmap.compress(Bitmap.CompressFormat.JPEG, 95, os) } else { bitmap.compress(Bitmap.CompressFormat.PNG, 100, os) } } return object : InputStreamOpener { override val mimeType : String get() = mime_type @Throws(IOException::class) override fun open() : InputStream { return FileInputStream(temp_file) } override fun deleteTempFile() { temp_file.delete() } } } finally { bitmap.recycle() } } } catch(ex : Throwable) { log.trace(ex) showToast(this, ex, "Resizing image failed.") } break } return object : InputStreamOpener { override val mimeType : String get() = mime_type @Throws(IOException::class) override fun open() : InputStream { return contentResolver.openInputStream(uri) ?: error("openInputStream returns null") } override fun deleteTempFile() { } } } private fun getMimeType(uri : Uri, mimeTypeArg : String?) : String? { // image/j()pg だの image/j(e)pg だの、mime type を誤記するアプリがあまりに多い // クレームで消耗するのを減らすためにファイルヘッダを確認する if(mimeTypeArg == null || mimeTypeArg.startsWith("image/")) { val sv = checkImageHeaderList(contentResolver, uri) if(sv != null) return sv } // 既に引数で与えられてる if(mimeTypeArg?.isNotEmpty() == true) { return mimeTypeArg } // ContentResolverに尋ねる var sv = contentResolver.getType(uri) if(sv?.isNotEmpty() == true) return sv // gboardのステッカーではUriのクエリパラメータにmimeType引数がある sv = uri.getQueryParameter("mimeType") if(sv?.isNotEmpty() == true) return sv return null } private fun checkAttachments(srcList : ArrayList?) { srcList?.forEach { addAttachment(it.uri, it.mimeType) } } private class AttachmentRequest( val account : SavedAccount, val pa : PostAttachment, val uri : Uri, val mimeType : String, val onUploadEnd : () -> Unit ) private val attachment_queue = ConcurrentLinkedQueue() private var attachment_worker : AttachmentWorker? = null private var lastAttachmentAdd : Long = 0L private var lastAttachmentComplete : Long = 0L @SuppressLint("StaticFieldLeak") private fun addAttachment( uri : Uri, mimeTypeArg : String? = null, onUploadEnd : () -> Unit = {} ) { if(attachment_list.size >= 4) { showToast(this, false, R.string.attachment_too_many) return } val account = this@ActPost.account if(account == null) { showToast(this, false, R.string.account_select_please) return } val mime_type = getMimeType(uri, mimeTypeArg) if(mime_type?.isEmpty() != false) { showToast(this, false, R.string.mime_type_missing) return } if(! acceptable_mime_types.contains(mime_type)) { showToast(this, true, R.string.mime_type_not_acceptable, mime_type) return } app_state.attachment_list = this.attachment_list val pa = PostAttachment(this) attachment_list.add(pa) showMediaAttachment() // アップロード開始トースト(連発しない) val now = System.currentTimeMillis() if(now - lastAttachmentAdd >= 5000L) { showToast(this, false, R.string.attachment_uploading) } lastAttachmentAdd = now // マストドンは添付メディアをID順に表示するため // 画像が複数ある場合は一つずつ処理する必要がある // 投稿画面ごとに1スレッドだけ作成してバックグラウンド処理を行う attachment_queue.add(AttachmentRequest(account, pa, uri, mime_type, onUploadEnd)) val oldWorker = attachment_worker if(oldWorker == null || ! oldWorker.isAlive || oldWorker.isInterrupted || oldWorker.isCancelled.get() ) { oldWorker?.cancel() attachment_worker = AttachmentWorker().apply { start() } } else { oldWorker.notifyEx() } } inner class AttachmentWorker : WorkerBase() { internal val isCancelled = AtomicBoolean(false) override fun cancel() { isCancelled.set(true) notifyEx() } override fun run() { try { while(! isCancelled.get()) { val item = attachment_queue.poll() if(item == null) { waitEx(86400000L) continue } val result = item.upload() handler.post { item.handleResult(result) } } } catch(ex : Throwable) { log.trace(ex) log.e(ex, "AttachmentWorker") } } private fun AttachmentRequest.upload() : TootApiResult? { if(mimeType.isEmpty()) { return TootApiResult("mime_type is empty.") } try { val client = TootApiClient(this@ActPost, callback = object : TootApiCallback { override val isApiCancelled : Boolean get() = isCancelled.get() }) client.account = account val opener = createOpener(uri, mimeType) val media_size_max = when { mimeType.startsWith("video") -> { 1000000 * Math.max(1, Pref.spMovieSizeMax.toInt(pref)) } else -> { 1000000 * Math.max(1, Pref.spMediaSizeMax.toInt(pref)) } } val content_length = getStreamSize(true, opener.open()) if(content_length > media_size_max) { return TootApiResult( getString( R.string.file_size_too_big, media_size_max / 1000000 ) ) } if(account.isMisskey) { val multipart_builder = MultipartBody.Builder() .setType(MultipartBody.FORM) val apiKey = account.token_info?.parseString(TootApiClient.KEY_API_KEY_MISSKEY) if(apiKey?.isNotEmpty() == true) { multipart_builder.addFormDataPart("i", apiKey) } multipart_builder.addFormDataPart( "file", getDocumentName(contentResolver, uri), object : RequestBody() { override fun contentType() : MediaType? { return MediaType.parse(opener.mimeType) } @Throws(IOException::class) override fun contentLength() : Long { return content_length } @Throws(IOException::class) override fun writeTo(sink : BufferedSink) { opener.open().use { inData -> val tmp = ByteArray(4096) while(true) { val r = inData.read(tmp, 0, tmp.size) if(r <= 0) break sink.write(tmp, 0, r) } } } } ) val result = client.request( "/api/drive/files/create", multipart_builder.build().toPost() ) opener.deleteTempFile() onUploadEnd() val jsonObject = result?.jsonObject if(jsonObject != null) { val a = parseItem(::TootAttachment, ServiceType.MISSKEY, jsonObject) if(a == null) { result.error = "TootAttachment.parse failed" } else { pa.attachment = a } } return result } else { val multipart_body = MultipartBody.Builder() .setType(MultipartBody.FORM) .addFormDataPart( "file", getDocumentName(contentResolver, uri), object : RequestBody() { override fun contentType() : MediaType? { return MediaType.parse(opener.mimeType) } @Throws(IOException::class) override fun contentLength() : Long { return content_length } @Throws(IOException::class) override fun writeTo(sink : BufferedSink) { opener.open().use { inData -> val tmp = ByteArray(4096) while(true) { val r = inData.read(tmp, 0, tmp.size) if(r <= 0) break sink.write(tmp, 0, r) } } } } ) val result = client.request( "/api/v1/media", multipart_body.build().toPost() ) opener.deleteTempFile() onUploadEnd() val jsonObject = result?.jsonObject if(jsonObject != null) { val a = parseItem(::TootAttachment, ServiceType.MASTODON, jsonObject) if(a == null) { result.error = "TootAttachment.parse failed" } else { pa.attachment = a } } return result } } catch(ex : Throwable) { return TootApiResult(ex.withCaption("read failed.")) } } private fun AttachmentRequest.handleResult(result : TootApiResult?) { if(pa.attachment == null) { pa.status = PostAttachment.STATUS_UPLOAD_FAILED if(result != null) { showToast(this@ActPost, true, result.error) } } else { pa.status = PostAttachment.STATUS_UPLOADED } // 投稿中に画面回転があった場合、新しい画面のコールバックを呼び出す必要がある pa.callback?.onPostAttachmentComplete(pa) } } // 添付メディア投稿が完了したら呼ばれる override fun onPostAttachmentComplete(pa : PostAttachment) { if(! attachment_list.contains(pa)) { // この添付メディアはリストにない return } when(pa.status) { PostAttachment.STATUS_UPLOAD_FAILED -> { // アップロード失敗 attachment_list.remove(pa) showMediaAttachment() } PostAttachment.STATUS_UPLOADED -> { val a = pa.attachment if(a != null) { // アップロード完了 val now = System.currentTimeMillis() if(now - lastAttachmentComplete >= 5000L) { showToast(this@ActPost, false, R.string.attachment_uploaded) } lastAttachmentComplete = now if(Pref.bpAppendAttachmentUrlToContent(pref)) { // 投稿欄の末尾に追記する val selStart = etContent.selectionStart val selEnd = etContent.selectionEnd val e = etContent.editableText val len = e.length val last_char = if(len <= 0) ' ' else e[len - 1] if(! CharacterGroup.isWhitespace(last_char.toInt())) { e.append(" ").append(a.text_url) } else { e.append(a.text_url) } etContent.setSelection(selStart, selEnd) } } showMediaAttachment() } else -> { // アップロード中…? } } } private fun performCamera() { try { // カメラで撮影 val filename = System.currentTimeMillis().toString() + ".jpg" val values = ContentValues() values.put(MediaStore.Images.Media.TITLE, filename) values.put(MediaStore.Images.Media.MIME_TYPE, "image/jpeg") uriCameraImage = contentResolver.insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values) val intent = Intent(MediaStore.ACTION_IMAGE_CAPTURE) intent.putExtra(MediaStore.EXTRA_OUTPUT, uriCameraImage) startActivityForResult(intent, REQUEST_CODE_CAMERA) } catch(ex : Throwable) { log.trace(ex) showToast(this, ex, "opening camera app failed.") } } private fun preparePermission() { if(Build.VERSION.SDK_INT >= 23) { // No explanation needed, we can request the permission. ActivityCompat.requestPermissions( this, arrayOf(Manifest.permission.WRITE_EXTERNAL_STORAGE) // Manifest.permission.CAMERA, , PERMISSION_REQUEST_CODE ) } else { showToast(this, true, R.string.missing_permission_to_access_media) } } override fun onRequestPermissionsResult( requestCode : Int, permissions : Array, grantResults : IntArray ) { when(requestCode) { PERMISSION_REQUEST_CODE -> { var bNotGranted = false var i = 0 val ie = permissions.size while(i < ie) { if(grantResults[i] != PackageManager.PERMISSION_GRANTED) { bNotGranted = true } ++ i } if(bNotGranted) { showToast(this, true, R.string.missing_permission_to_access_media) } else { openAttachment() } } } } private fun showVisibility() = setIconDrawableId( this, btnVisibility, Styler.getVisibilityIconId( account?.isMisskey == true , visibility ?: TootVisibility.Public ), getAttributeColor(this, R.attr.colorVectorDrawable) ) private fun performVisibility() { val list = if(account?.isMisskey == true) { arrayOf( // TootVisibility.WebSetting, TootVisibility.Public, TootVisibility.UnlistedHome, TootVisibility.PrivateFollowers, TootVisibility.LocalPublic, TootVisibility.LocalHome, TootVisibility.LocalFollowers, TootVisibility.DirectSpecified, TootVisibility.DirectPrivate ) } else { arrayOf( TootVisibility.WebSetting, TootVisibility.Public, TootVisibility.UnlistedHome, TootVisibility.PrivateFollowers, TootVisibility.DirectSpecified ) } val caption_list = list .map { Styler.getVisibilityCaption(this, account?.isMisskey == true, it) } .toTypedArray() AlertDialog.Builder(this) .setTitle(R.string.choose_visibility) .setItems(caption_list) { _, which -> if(which in 0 until list.size) { visibility = list[which] showVisibility() } } .setNegativeButton(R.string.cancel, null) .show() } /////////////////////////////////////////////////////////////////////////////////////// private fun performMore() { val dialog = ActionsDialog() dialog.addAction(getString(R.string.open_picker_emoji)) { post_helper.openEmojiPickerFromMore() } dialog.addAction(getString(R.string.clear_text)) { etContent.setText("") etContentWarning.setText("") } dialog.addAction(getString(R.string.clear_text_and_media)) { etContent.setText("") etContentWarning.setText("") attachment_list.clear() showMediaAttachment() } if(PostDraft.hasDraft()) dialog.addAction(getString(R.string.restore_draft)) { openDraftPicker() } dialog.addAction(getString(R.string.recommended_plugin)) { showRecommendedPlugin(null) } dialog.show(this, null) } /////////////////////////////////////////////////////////////////////////////////////// // post private fun performPost() { val account = this.account ?: return // アップロード中は投稿できない for(pa in attachment_list) { if(pa.status == PostAttachment.STATUS_UPLOADING) { showToast(this, false, R.string.media_attachment_still_uploading) return } } post_helper.content = etContent.text.toString().trim { it <= ' ' } if(! cbEnquete.isChecked) { post_helper.enquete_items = null } else { val enquete_items = ArrayList() for(et in list_etChoice) { enquete_items.add(et.text.toString().trim { it <= ' ' }) } post_helper.enquete_items = enquete_items } if(! cbContentWarning.isChecked) { post_helper.spoiler_text = null // nullはCWチェックなしを示す } else { post_helper.spoiler_text = etContentWarning.text.toString().trim { it <= ' ' } } post_helper.visibility = this.visibility ?: TootVisibility.Public post_helper.bNSFW = cbNSFW.isChecked post_helper.in_reply_to_id = this.in_reply_to_id post_helper.attachment_list = this.attachment_list post_helper.emojiMapCustom = App1.custom_emoji_lister.getMap(account.host, account.isMisskey) post_helper.redraft_status_id = redraft_status_id post_helper.useQuotedRenote = cbQuoteRenote.isChecked post_helper.scheduledAt = timeSchedule post_helper.scheduledId = scheduledStatus?.id post_helper.post(account, callback = object : PostHelper.PostCompleteCallback { override fun onPostComplete( target_account : SavedAccount, status : TootStatus ) { val data = Intent() data.putExtra(EXTRA_POSTED_ACCT, target_account.acct) status.id.putTo(data, EXTRA_POSTED_STATUS_ID) redraft_status_id?.putTo(data, EXTRA_POSTED_REDRAFT_ID) status.in_reply_to_id?.putTo(data, EXTRA_POSTED_REPLY_ID) setResult(RESULT_OK, data) isPostComplete = true this@ActPost.finish() } override fun onScheduledPostComplete(target_account : SavedAccount) { showToast(this@ActPost, false, getString(R.string.scheduled_status_sent)) val data = Intent() data.putExtra(EXTRA_POSTED_ACCT, target_account.acct) setResult(RESULT_OK, data) isPostComplete = true this@ActPost.finish() } }) } private fun showQuotedRenote() { val isReply = in_reply_to_id != null val isMisskey = account?.isMisskey == true cbQuoteRenote.visibility = if(isReply && isMisskey) View.VISIBLE else View.GONE } internal fun showReplyTo() { if(in_reply_to_id == null) { llReply.visibility = View.GONE } else { llReply.visibility = View.VISIBLE tvReplyTo.text = DecodeOptions( this@ActPost, linkHelper = account, short = true, decodeEmoji = true ).decodeHTML(in_reply_to_text) ivReply.setImageUrl(pref, Styler.calcIconRound(ivReply.layoutParams), in_reply_to_image) } } private fun removeReply() { in_reply_to_id = null in_reply_to_text = null in_reply_to_image = null in_reply_to_url = null showReplyTo() showQuotedRenote() } private fun saveDraft() { val content = etContent.text.toString() val content_warning = if(cbContentWarning.isChecked) etContentWarning.text.toString() else "" val isEnquete = cbEnquete.isChecked val str_choice = arrayOf( if(isEnquete) list_etChoice[0].text.toString() else "", if(isEnquete) list_etChoice[1].text.toString() else "", if(isEnquete) list_etChoice[2].text.toString() else "", if(isEnquete) list_etChoice[3].text.toString() else "" ) var hasContent = false if(content.isNotBlank()) hasContent = true if(content_warning.isNotBlank()) hasContent = true for(s in str_choice) { if(s.isNotBlank()) hasContent = true } if(! hasContent) { log.d("saveDraft: dont save empty content") return } try { val tmp_attachment_list = JSONArray() for(pa in attachment_list) { val json = pa.attachment?.encodeJson() if(json != null) tmp_attachment_list.put(json) } val json = JSONObject() json.put(DRAFT_CONTENT, content) json.put(DRAFT_CONTENT_WARNING, content_warning) json.put(DRAFT_CONTENT_WARNING_CHECK, cbContentWarning.isChecked) json.put(DRAFT_NSFW_CHECK, cbNSFW.isChecked) visibility?.let { json.put(DRAFT_VISIBILITY, it.id.toString()) } json.put(DRAFT_ACCOUNT_DB_ID, account?.db_id ?: - 1L) json.put(DRAFT_ATTACHMENT_LIST, tmp_attachment_list) in_reply_to_id?.putTo(json, DRAFT_REPLY_ID) json.put(DRAFT_REPLY_TEXT, in_reply_to_text) json.put(DRAFT_REPLY_IMAGE, in_reply_to_image) json.put(DRAFT_REPLY_URL, in_reply_to_url) json.put(DRAFT_QUOTED_RENOTE, cbQuoteRenote.isChecked) json.put(DRAFT_IS_ENQUETE, isEnquete) val array = JSONArray() for(s in str_choice) { array.put(s) } json.put(DRAFT_ENQUETE_ITEMS, array) PostDraft.save(System.currentTimeMillis(), json) } catch(ex : Throwable) { log.trace(ex) } } private fun openDraftPicker() { DlgDraftPicker().open(this) { draft -> restoreDraft(draft) } } private fun restoreDraft(draft : JSONObject) { @Suppress("DEPRECATION") val progress = ProgressDialogEx(this) val task = @SuppressLint("StaticFieldLeak") object : AsyncTask() { val list_warning = ArrayList() var account : SavedAccount? = null override fun doInBackground(vararg params : Void) : String? { var content = draft.parseString(DRAFT_CONTENT) ?: "" val account_db_id = draft.parseLong(DRAFT_ACCOUNT_DB_ID) ?: - 1L val tmp_attachment_list = draft.optJSONArray(DRAFT_ATTACHMENT_LIST)?.toObjectList() val account = SavedAccount.loadAccount(this@ActPost, account_db_id) if(account == null) { list_warning.add(getString(R.string.account_in_draft_is_lost)) try { if(tmp_attachment_list != null) { // 本文からURLを除去する tmp_attachment_list.forEach { val text_url = TootAttachment.decodeJson(it).text_url if(text_url?.isNotEmpty() == true) { content = content.replace(text_url, "") } } tmp_attachment_list.clear() draft.put(DRAFT_ATTACHMENT_LIST, tmp_attachment_list.toJsonArray()) draft.put(DRAFT_CONTENT, content) draft.remove(DRAFT_REPLY_ID) draft.remove(DRAFT_REPLY_TEXT) draft.remove(DRAFT_REPLY_IMAGE) draft.remove(DRAFT_REPLY_URL) } } catch(ignored : JSONException) { } return "OK" } this.account = account // アカウントがあるなら基本的にはすべての情報を復元できるはずだが、いくつか確認が必要だ val api_client = TootApiClient(this@ActPost, callback = object : TootApiCallback { override val isApiCancelled : Boolean get() = isCancelled override fun publishApiProgress(s : String) { runOnMainLooper { progress.setMessage(s) } } }) api_client.account = account if(in_reply_to_id != null) { val result = api_client.request("/api/v1/statuses/$in_reply_to_id") if(isCancelled) return null val jsonObject = result?.jsonObject if(jsonObject == null) { list_warning.add(getString(R.string.reply_to_in_draft_is_lost)) draft.remove(DRAFT_REPLY_ID) draft.remove(DRAFT_REPLY_TEXT) draft.remove(DRAFT_REPLY_IMAGE) } } try { if(tmp_attachment_list != null) { // 添付メディアの存在確認 var isSomeAttachmentRemoved = false val it = tmp_attachment_list.iterator() while(it.hasNext()) { if(isCancelled) return null val ta = TootAttachment.decodeJson(it.next()) if(check_exist(ta.url)) continue it.remove() isSomeAttachmentRemoved = true // 本文からURLを除去する val text_url = ta.text_url if(text_url?.isNotEmpty() == true) { content = content.replace(text_url, "") } } if(isSomeAttachmentRemoved) { list_warning.add(getString(R.string.attachment_in_draft_is_lost)) draft.put(DRAFT_ATTACHMENT_LIST, tmp_attachment_list.toJsonArray()) draft.put(DRAFT_CONTENT, content) } } } catch(ex : JSONException) { log.trace(ex) } return "OK" } override fun onCancelled(result : String?) { onPostExecute(result) } override fun onPostExecute(result : String?) { progress.dismiss() if(isCancelled || result == null) { // cancelled. return } val content = draft.optString(DRAFT_CONTENT) val content_warning = draft.optString(DRAFT_CONTENT_WARNING) val content_warning_checked = draft.optBoolean(DRAFT_CONTENT_WARNING_CHECK) val nsfw_checked = draft.optBoolean(DRAFT_NSFW_CHECK) val tmp_attachment_list = draft.optJSONArray(DRAFT_ATTACHMENT_LIST) val reply_id = EntityId.from(draft, DRAFT_REPLY_ID) val reply_text = draft.optString(DRAFT_REPLY_TEXT, null) val reply_image = draft.optString(DRAFT_REPLY_IMAGE, null) val reply_url = draft.optString(DRAFT_REPLY_URL, null) val draft_visibility = TootVisibility .parseSavedVisibility(draft.parseString(DRAFT_VISIBILITY)) val evEmoji = DecodeOptions(this@ActPost, decodeEmoji = true).decodeEmoji(content) etContent.setText(evEmoji) etContent.setSelection(evEmoji.length) etContentWarning.setText(content_warning) etContentWarning.setSelection(content_warning.length) cbContentWarning.isChecked = content_warning_checked cbNSFW.isChecked = nsfw_checked if(draft_visibility != null) this@ActPost.visibility = draft_visibility cbQuoteRenote.isChecked = draft.optBoolean(DRAFT_QUOTED_RENOTE) cbEnquete.isChecked = draft.optBoolean(DRAFT_IS_ENQUETE, false) val array = draft.optJSONArray(DRAFT_ENQUETE_ITEMS) if(array != null) { var src_index = 0 for(et in list_etChoice) { if(src_index < array.length()) { et.setText(array.optString(src_index)) ++ src_index } else { et.setText("") } } } if(account != null) selectAccount(account) if(tmp_attachment_list.length() > 0) { attachment_list.clear() tmp_attachment_list.forEach { if(it !is JSONObject) return@forEach val pa = PostAttachment(TootAttachment.decodeJson(it)) attachment_list.add(pa) } } if(reply_id != null) { in_reply_to_id = reply_id in_reply_to_text = reply_text in_reply_to_image = reply_image in_reply_to_url = reply_url } updateContentWarning() showMediaAttachment() showVisibility() updateTextCount() showReplyTo() showEnquete() showQuotedRenote() if(! list_warning.isEmpty()) { val sb = StringBuilder() for(s in list_warning) { if(sb.isNotEmpty()) sb.append("\n") sb.append(s) } AlertDialog.Builder(this@ActPost) .setMessage(sb) .setNeutralButton(R.string.close, null) .show() } } } progress.isIndeterminate = true progress.setCancelable(true) progress.setOnCancelListener { task.cancel(true) } progress.show() task.executeOnExecutor(App1.task_executor) } private fun prepareMushroomText(et : EditText) : String { mushroom_start = et.selectionStart mushroom_end = et.selectionEnd return if(mushroom_end > mushroom_start) { et.text.toString().substring(mushroom_start, mushroom_end) } else { "" } } private fun applyMushroomText(et : EditText, text : String) { val src = et.text.toString() if(mushroom_start > src.length) mushroom_start = src.length if(mushroom_end > src.length) mushroom_end = src.length val sb = StringBuilder() sb.append(src.substring(0, mushroom_start)) // int new_sel_start = sb.length(); sb.append(text) val new_sel_end = sb.length sb.append(src.substring(mushroom_end)) et.setText(sb) et.setSelection(new_sel_end, new_sel_end) } private fun openMushroom() { try { var text : String? = null when { etContentWarning.hasFocus() -> { mushroom_input = 1 text = prepareMushroomText(etContentWarning) } etContent.hasFocus() -> { mushroom_input = 0 text = prepareMushroomText(etContent) } else -> for(i in 0 .. 3) { if(list_etChoice[i].hasFocus()) { mushroom_input = i + 2 text = prepareMushroomText(list_etChoice[i]) } } } if(text == null) { mushroom_input = 0 text = prepareMushroomText(etContent) } val intent = Intent("com.adamrocker.android.simeji.ACTION_INTERCEPT") intent.addCategory("com.adamrocker.android.simeji.REPLACE") intent.putExtra("replace_key", text) // Create intent to show chooser val chooser = Intent.createChooser(intent, getString(R.string.select_plugin)) // Verify the intent will resolve to at least one activity if(intent.resolveActivity(packageManager) == null) { showRecommendedPlugin(getString(R.string.plugin_not_installed)) return } startActivityForResult(chooser, REQUEST_CODE_MUSHROOM) } catch(ex : Throwable) { log.trace(ex) showRecommendedPlugin(getString(R.string.plugin_not_installed)) } } private fun applyMushroomResult(text : String) { when(mushroom_input) { 0 -> applyMushroomText(etContent, text) 1 -> applyMushroomText(etContentWarning, text) else -> for(i in 0 .. 3) { if(mushroom_input == i + 2) { applyMushroomText(list_etChoice[i], text) } } } } @SuppressLint("InflateParams") private fun showRecommendedPlugin(title : String?) { val res_id = when(getString(R.string.language_code)) { "ja" -> R.raw.recommended_plugin_ja "fr" -> R.raw.recommended_plugin_fr else -> R.raw.recommended_plugin_en } this.loadRawResource(res_id).let { data -> val text = data.decodeUTF8() val viewRoot = layoutInflater.inflate(R.layout.dlg_plugin_missing, null, false) val tvText = viewRoot.findViewById(R.id.tvText) val sv = DecodeOptions(this@ActPost, LinkHelper.nullHost).decodeHTML(text) tvText.text = sv tvText.movementMethod = LinkMovementMethod.getInstance() val tvTitle = viewRoot.findViewById(R.id.tvTitle) if(title?.isEmpty() != false) { tvTitle.visibility = View.GONE } else { tvTitle.text = title } AlertDialog.Builder(this) .setView(viewRoot) .setCancelable(true) .setNeutralButton(R.string.close, null) .show() } } private fun showEnquete() { llEnquete.visibility = if(cbEnquete.isChecked) View.VISIBLE else View.GONE } private val commitContentListener = InputConnectionCompat.OnCommitContentListener { inputContentInfo : InputContentInfoCompat, flags : Int, _ : Bundle? -> // Intercepts InputConnection#commitContent API calls. // - inputContentInfo : content to be committed // - flags : {@code 0} or {@link #INPUT_CONTENT_GRANT_READ_URI_PERMISSION} // - opts : optional bundle data. This can be {@code null} // return // - true if this request is accepted by the application, // no matter if the request is already handled or still being handled in background. // - false to use the default implementation // read and display inputContentInfo asynchronously if(Build.VERSION.SDK_INT >= 25 && flags and InputConnectionCompat.INPUT_CONTENT_GRANT_READ_URI_PERMISSION != 0 ) { try { inputContentInfo.requestPermission() } catch(e : Exception) { return@OnCommitContentListener false // return false if failed } } addAttachment(inputContentInfo.contentUri) { inputContentInfo.releasePermission() } true } private fun showSchedule() { tvSchedule.text = when(timeSchedule) { 0L -> getString(R.string.unspecified) else -> TootStatus.formatTime(this, timeSchedule, true) } } private fun performSchedule() { DlgDateTime(this).open(timeSchedule) { t -> timeSchedule = t showSchedule() } } private fun resetSchedule() { timeSchedule = 0L showSchedule() } }