Загрузить изображения в Imgur из Mathematica - PullRequest
52 голосов
/ 07 ноября 2011

Вот вызов для всех mathematica подписчиков тегов. Давайте сделаем намного удобнее вставку изображений в SO сообщение из Mathematica, создав загрузчик imgur .

Как мы можем создать функцию imgur[g_], которая будет растеризовать свой аргумент (убедившись, что окончательный размер не шире ширины сообщений StackOverflow), преобразовать его в PNG, загрузить в imgur и вернуть готовый быть вставленным MarkDown строка, такая как ![Mathematica graphic](http://i.imgur.com/ZENa4.jpg)?

Полезные ссылки:

Мне не удалось адаптировать этот последний метод для загрузки изображения без предварительного экспорта в файл.


Предупреждение, используйте с осторожностью! StackOverflow использует отдельную установку imgur , которая хранит изображения в течение неопределенного времени. Если вы используете основной imgur, изображения исчезнут через 6 месяцев, если их никто не просматривает . К сожалению, по состоянию на ноябрь 2011 года, по-видимому, нет официального способа программной загрузки изображений в StackOverflow.


Обновление: См. Ниже решение для загрузки в StackOverflow напрямую .

Ответы [ 3 ]

16 голосов
/ 01 декабря 2011

Маленькая птичка только что сообщила мне о Mathematica решении этого вопроса (базовая реализация все еще использует JLink, но этот ответ скрывает весь код, связанный с Java):

imgur[expr_] := Module[
 {url, key, image, data, xml, imgurUrl},
 url = "http://api.imgur.com/2/upload";
 key = "c07bc3fb59ef878d5e23a0c4972fbb29";
 image = Fold[ExportString, expr, {"PNG", "Base64"}];
 xml = Import[url, 
  "XML", "RequestMethod" -> "POST", 
  "RequestParameters" -> {"key" -> key, "image" -> image}];
 imgurUrl = Cases[xml, XMLElement["original", {}, {string_}] :> string, 
  Infinity][[1]];
 "![Mathematica graphic](" <> imgurUrl <> ")"
]

Это только V8, а параметры импорта XML "RequestMethod" и "RequestParameters" не имеют документов и являются экспериментальными (и поэтому могут быть изменены).

13 голосов
/ 01 декабря 2011

Примечание: Получите готовую палитру с этой функциональностью здесь .


Решение от Arnoud взволновало меня и привело к нетерпению, так что вот улучшениеЭто.Я не мог бы сделать это без изучения его кода.Эта версия кажется несколько более надежной и менее подверженной ошибкам тайм-аута, но, если честно, я вообще не знаю Java, поэтому любые улучшения приветствуются.

Самое главное: эта версия загружается непосредственно в stack.imgur.comтак что здесь безопасно использовать StackOverflow, не беспокоясь о том, что загруженные изображения исчезнут через некоторое время.

Я предоставляю три функции:

  • stackImage загружает выражение,экспортируется как PNG и возвращает URL
  • stackMarkdown возвращает уценку, готовую к копированию
  • stackCopyMarkdown копирует уценку в буфер обмена

Далеешаг: создайте кнопку палитры, которая делает это автоматически для выбранной графики в блокноте.Улучшения кода очень приветствуются.


Needs["JLink`"]


stackImage::httperr = "Server returned respose code: `1`";
stackImage::err = "Server returner error: `1`";

stackImage[g_] :=
 Module[
  {getVal, url, client, method, data, partSource, part, entity, code, 
   response, error, result},

  (* this function attempts to parse the response fro the SO server *)
  getVal[res_, key_String] :=
   With[{k = "var " <> key <> " = "},
    StringTrim[
     First@StringCases[First@Select[res, StringMatchQ[#, k ~~ ___] &], 
       k ~~ v___ ~~ ";" :> v],
     "'"]
    ];

  data = ExportString[g, "PNG"];

  JavaBlock[
    url = "https://stackoverflow.com/upload/image";
    client = JavaNew["org.apache.commons.httpclient.HttpClient"];
    method = JavaNew["org.apache.commons.httpclient.methods.PostMethod", url];
    partSource = JavaNew["org.apache.commons.httpclient.methods.multipart.ByteArrayPartSource", "mmagraphics.png", MakeJavaObject[data]@toCharArray[]];
    part = JavaNew["org.apache.commons.httpclient.methods.multipart.FilePart", "name", partSource];
    part@setContentType["image/png"];
    entity = JavaNew["org.apache.commons.httpclient.methods.multipart.MultipartRequestEntity", {part}, method@getParams[]];
    method@setRequestEntity[entity];
    code = client@executeMethod[method];
    response = method@getResponseBodyAsString[];
  ]

  If[code =!= 200, Message[stackImage::httperr, code]; Return[$Failed]];
  response = StringTrim /@ StringSplit[response, "\n"];

  error = getVal[response, "error"];
  result = getVal[response, "result"];
  If[StringMatchQ[result, "http*"],
   result,
   Message[stackImage::err, error]; $Failed]
  ]


stackMarkdown[g_] := "![Mathematica graphics](" <> stackImage[g] <> ")"


stackCopyMarkdown[g_] := Module[{nb, markdown},
  markdown = Check[stackMarkdown[g], $Failed];
  If[markdown =!= $Failed,
   nb = NotebookCreate[Visible -> False];
   NotebookWrite[nb, Cell[markdown, "Text"]];
   SelectionMove[nb, All, Notebook];
   FrontEndTokenExecute[nb, "Copy"];
   NotebookClose[nb];
   ]
  ]

Обновление:

Вот кнопка, которая покажет предварительный просмотр выбора и предложит выгрузку (или отмена).Требуется определить предыдущие функции.

Button["Upload to SO",
 Module[{cell = NotebookRead@InputNotebook[], img},
  If[cell =!= {}, img = Rasterize[cell];
   MessageDialog[
    Column[{"Upload image to StackExchange sites?", 
      img}], {"Upload and copy MarkDown" :> stackCopyMarkdown[img], 
     "Cancel" :> Null}, WindowTitle -> "Upload to StackExchange"]]]]

К сожалению, я не могу поместить кнопку в палитру (CreatePalette), потому что размеры палитры будут влиять на растеризацию.Решения этой проблемы приветствуются.

Обновление 2:

Основываясь на ответе на этот вопрос , вот рабочая кнопка палитры только для Windows:

button = Button["Upload to SO",
  Module[{sel},
   FrontEndExecute[
    FrontEndToken[FrontEnd`SelectedNotebook[], "CopySpecial", "MGF"]];
   sel = Cases[NotebookGet@ClipboardNotebook[], 
     RasterBox[data_, ___] :> 
      Image[data, "Byte", ColorSpace -> "RGB", Magnification -> 1], 
     Infinity];
   If[sel =!= {},
    With[{img = First[sel]},
     MessageDialog[
      Column[{"Upload image to StackExchange sites?", 
        img}], {"Upload and copy MarkDown" :> stackCopyMarkdown[img], 
       "Cancel" :> Null}, WindowTitle -> "Upload to StackExchange"]
     ]
    ]
   ]
  ]

CreatePalette[button]

Предупреждение: уничтожает содержимое буфера обмена, даже если нажать кнопку «Отмена» в окне предварительного просмотра.

12 голосов
/ 01 декабря 2011

Примечание : используется анонимный загрузчик imgur с моим анонимным ключом. Сайт imgur ограничивает загрузку до 50 загрузок в час, что должно быть нормально, но это может вызвать проблемы, если многие люди пробуют это одновременно. Поэтому, пожалуйста, получите свой собственный анонимный ключ здесь:

http://imgur.com/register/api_anon

А затем замените ключ в приведенном ниже коде своим собственным ключом ( спасибо! ).

Самой сложной частью кода было преобразование выражения Mathematica в изображение PNG в кодировку Base64 в кодировку URL. Есть около 1000 способов сделать это неправильно, и я думаю, что мне удалось попробовать их все.

Код разбивается на несколько частей:

  • Постройте URL-адрес POST
  • Установите соединение HTTP
  • Отправьте ПОЧТУ URL
  • Считать результат, который является XML
  • Извлечение imgur url из XML
  • Отформатируйте URL-адрес imgur как уценку (или как функцию Mathematica Hyperlink).

Вот код:

imgur[expr_] :=
 Module[{url, key, image, data, jUrl, jConn, jWriter, jInput, buffer,
   byte, xml, imgurUrl},
  Needs["JLink`"];
  JLink`JavaBlock[
   JLink`LoadJavaClass["java.net.URLEncoder"];
   url = "http://api.imgur.com/2/upload";
   key = "c07bc3fb59ef878d5e23a0c4972fbb29";
   image = ExportString[ExportString[expr, "PNG"], "Base64"];
   data =
    URLEncoder`encode["key"   , "UTF-8"] <> "=" <>
    URLEncoder`encode[ key    , "UTF-8"] <> "&" <>
    URLEncoder`encode["image" , "UTF-8"] <> "=" <>
    URLEncoder`encode[ image  , "UTF-8"] ;
   jUrl = JLink`JavaNew["java.net.URL", url];
   jConn = jUrl@openConnection[];
   jConn@setDoOutput[True];
   jWriter =
    JLink`JavaNew["java.io.OutputStreamWriter",
     jConn@getOutputStream[]];
   jWriter@write[data];
   jWriter@flush[];
   jInput = jConn@getInputStream[];
   buffer = {};
   While[(byte = jInput@read[]; byte >= 0), AppendTo[buffer, byte]];
   ];
  xml = ImportString[FromCharacterCode[buffer], "XML"];
  imgurUrl =
   Cases[xml,
     XMLElement["original", {}, {string_}] :>
      string, \[Infinity]][[1]];
  "![Mathematica graphic](" <> imgurUrl <> ")"
  ]

Тестирование:

In[]:= g = Graphics[{Blue, Disk[]}, PlotRange -> 1.2, ImageSize -> Small];
       pic = Overlay[{Blur[Binarize@g, 10], g}];
       imgur[pic]

Out[]= ![Mathematica graphic](http://i.imgur.com/eGOlL.png)

и фактическое изображение:

Mathematica graphic

...