1 // OpenTween - Client of Twitter
2 // Copyright (c) 2007-2011 kiri_feather (@kiri_feather) <kiri.feather@gmail.com>
3 // (c) 2008-2011 Moz (@syo68k)
4 // (c) 2008-2011 takeshik (@takeshik) <http://www.takeshik.org/>
5 // (c) 2010-2011 anis774 (@anis774) <http://d.hatena.ne.jp/anis774/>
6 // (c) 2010-2011 fantasticswallow (@f_swallow) <http://twitter.com/f_swallow>
7 // (c) 2011 spinor (@tplantd) <http://d.hatena.ne.jp/spinor/>
8 // (c) 2014 kim_upsilon (@kim_upsilon) <https://upsilo.net/~upsilon/>
9 // All rights reserved.
11 // This file is part of OpenTween.
13 // This program is free software; you can redistribute it and/or modify it
14 // under the terms of the GNU General Public License as published by the Free
15 // Software Foundation; either version 3 of the License, or (at your option)
18 // This program is distributed in the hope that it will be useful, but
19 // WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
20 // or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
23 // You should have received a copy of the GNU General Public License along
24 // with this program. If not, see <http://www.gnu.org/licenses/>, or write to
25 // the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor,
26 // Boston, MA 02110-1301, USA.
29 using System.Collections.Generic;
31 using System.Drawing.Imaging;
34 using System.Threading.Tasks;
35 using OpenTween.Api.DataModel;
36 using OpenTween.Setting;
38 namespace OpenTween.Connection
40 public class TwitterPhoto : IMediaUploadService
42 private readonly string[] pictureExt = { ".jpg", ".jpeg", ".gif", ".png" };
44 private readonly Twitter tw;
45 private TwitterConfiguration twitterConfig;
47 public TwitterPhoto(Twitter twitter, TwitterConfiguration twitterConfig)
50 this.twitterConfig = twitterConfig;
53 public int MaxMediaCount => 4;
55 public string SupportedFormatsStrForDialog => "Image Files(*.gif;*.jpg;*.jpeg;*.png)|*.gif;*.jpg;*.jpeg;*.png";
57 public bool CanUseAltText => true;
59 public bool CheckFileExtension(string fileExtension)
60 => this.pictureExt.Contains(fileExtension, StringComparer.InvariantCultureIgnoreCase);
62 public bool CheckFileSize(string fileExtension, long fileSize)
64 var maxFileSize = this.GetMaxFileSize(fileExtension);
65 return maxFileSize == null || fileSize <= maxFileSize.Value;
68 public long? GetMaxFileSize(string fileExtension)
69 => this.twitterConfig.PhotoSizeLimit;
71 public async Task<PostStatusParams> UploadAsync(IMediaItem[] mediaItems, PostStatusParams postParams)
73 if (mediaItems == null)
74 throw new ArgumentNullException(nameof(mediaItems));
76 if (mediaItems.Length == 0)
77 throw new ArgumentException("Err:Media not specified.");
79 foreach (var item in mediaItems)
82 throw new ArgumentException("Err:Media not specified.");
85 throw new ArgumentException("Err:Media not found.");
90 if (Twitter.DMSendTextRegex.IsMatch(postParams.Text))
91 mediaIds = new[] { await this.UploadMediaForDM(mediaItems).ConfigureAwait(false) };
93 mediaIds = await this.UploadMediaForTweet(mediaItems).ConfigureAwait(false);
95 postParams.MediaIds = mediaIds;
100 // pic.twitter.com の URL は文字数にカウントされない
101 public int GetReservedTextLength(int mediaCount)
104 public void UpdateTwitterConfiguration(TwitterConfiguration config)
105 => this.twitterConfig = config;
107 private async Task<long[]> UploadMediaForTweet(IMediaItem[] mediaItems)
109 var uploadTasks = from m in mediaItems
110 select this.UploadMediaItem(m, mediaCategory: null);
112 var mediaIds = await Task.WhenAll(uploadTasks)
113 .ConfigureAwait(false);
118 private async Task<long> UploadMediaForDM(IMediaItem[] mediaItems)
120 if (mediaItems.Length > 1)
121 throw new InvalidOperationException("Err:Can't attach multiple media to DM.");
123 var mediaItem = mediaItems[0];
125 string mediaCategory;
126 switch (mediaItem.Extension)
129 mediaCategory = "dm_gif";
132 mediaCategory = "dm_image";
136 var mediaId = await this.UploadMediaItem(mediaItems[0], mediaCategory)
137 .ConfigureAwait(false);
142 private async Task<long> UploadMediaItem(IMediaItem mediaItem, string mediaCategory)
144 async Task<long> UploadInternal(IMediaItem media, string category)
146 var mediaId = await this.tw.UploadMedia(media, category)
147 .ConfigureAwait(false);
149 if (!string.IsNullOrEmpty(media.AltText))
151 await this.tw.Api.MediaMetadataCreate(mediaId, media.AltText)
152 .ConfigureAwait(false);
158 using var origImage = mediaItem.CreateImage();
160 if (SettingManager.Common.AlphaPNGWorkaround && this.AddAlphaChannelIfNeeded(origImage.Image, out var newImage))
162 using var newMediaItem = new MemoryImageMediaItem(newImage);
163 newMediaItem.AltText = mediaItem.AltText;
165 return await UploadInternal(newMediaItem, mediaCategory);
169 return await UploadInternal(mediaItem, mediaCategory);
174 /// pic.twitter.com アップロード時に JPEG への変換を回避するための加工を行う
177 /// pic.twitter.com へのアップロード時に、アルファチャンネルを持たない PNG 画像が
178 /// JPEG 形式に変換され画質が低下する問題を回避します。
179 /// PNG 以外の画像や、すでにアルファチャンネルを持つ PNG 画像に対しては何もしません。
181 /// <returns>加工が行われた場合は true、そうでない場合は false</returns>
182 private bool AddAlphaChannelIfNeeded(Image origImage, out MemoryImage newImage)
186 // PNG 画像以外に対しては何もしない
187 if (origImage.RawFormat.Guid != ImageFormat.Png.Guid)
190 using var bitmap = new Bitmap(origImage);
192 // アルファ値が 255 以外のピクセルが含まれていた場合は何もしない
193 foreach (var x in Enumerable.Range(0, bitmap.Width))
195 foreach (var y in Enumerable.Range(0, bitmap.Height))
197 if (bitmap.GetPixel(x, y).A != 255)
202 // 左上の 1px だけアルファ値を 254 にする
203 var pixel = bitmap.GetPixel(0, 0);
204 var newPixel = Color.FromArgb(pixel.A - 1, pixel.R, pixel.G, pixel.B);
205 bitmap.SetPixel(0, 0, newPixel);
207 // MemoryImage 作成時に画像はコピーされるため、この後 bitmap は破棄しても問題ない
208 newImage = MemoryImage.CopyFromImage(bitmap);