2017-09-06 165 views
-1

我可以在後臺發送電子郵件,但無法弄清楚爲什麼使用「Mail.java」類中的addAttachment()方法不起作用。我到處尋找答案,但只找到了「相似」的答案,沒有幫助。我想要的只是從我的res/drawable文件夾發送圖像。發送電子郵件(在後臺)工作,添加附件不

我目前從steveholt55's Github借用代碼嘗試獲取附件發送..但仍然沒有運氣。到目前爲止,電子郵件在我的活動中沒有添加addAttachment()行,這是個好消息。但我也需要這個附件才能發送。

Mail.Java文件在下面,但我認爲這個問題是在我的代碼中的某處..特別是我試圖在我的活動中添加附件。 addAttachment()方法只請求一個字符串值。我想知道我是否在給它錯誤的道路?從我的活動

代碼:

if(!sent){ 
    try{ 

     String[] recipients = {"[email protected]"}; 
     SendEmailAsyncTask email = new SendEmailAsyncTask(); 
     email.m = new Mail(Config.EMAIL,Config.PASSWORD); 
     email.m.set_from("[email protected]"); 
     email.m.setBody("Body Text"); 
     email.m.set_subject("Subject Text"); 
     email.m.set_to(recipients); 
     email.m.addAttachment("res/drawable-hdpi/add_down.png"); 
     email.execute(); 

    } catch(Exception e){e.printStackTrace();} 
} 

這裏是異步類:

class SendEmailAsyncTask extends AsyncTask<Void, Void, Boolean> { 
    Mail m; 

    public SendEmailAsyncTask() { } 

    @Override 
    protected Boolean doInBackground(Void... params) { 
     boolean returnvalue = false; 

     try { 
      if (m.send()) 
       returnvalue = true; 
     } catch (Exception e) { 
      e.printStackTrace(); 
     } 

     return returnvalue; 
    } 
} 

下面是Mail.Java類:

public class Mail extends javax.mail.Authenticator { 
    private String _user; 
    private String _pass; 

    private String[] _to; 
    private String _from; 

    private String _port; 
    private String _sport; 

    private String _host; 

    private String _subject; 
    private String _body; 

    private boolean _auth; 

    private boolean _debuggable; 

    private Multipart _multipart; 

    public Mail() { 
     _host = "smtp.gmail.com"; // default smtp server 
     _port = "465"; // default smtp port 
     _sport = "465"; // default socketfactory port 

     _user = ""; // username 
     _pass = ""; // password 
     _from = ""; // email sent from 
     _subject = ""; // email subject 
     _body = ""; // email body 

     _debuggable = false; // debug mode on or off - default off 
     _auth = true; // smtp authentication - default on 

     _multipart = new MimeMultipart(); 

     // There is something wrong with MailCap, javamail can not find a 
     // handler for the multipart/mixed part, so this bit needs to be added. 
     MailcapCommandMap mc = (MailcapCommandMap) CommandMap 
       .getDefaultCommandMap(); 
     mc.addMailcap("text/html;; x-java-content-handler=com.sun.mail.handlers.text_html"); 
     mc.addMailcap("text/xml;; x-java-content-handler=com.sun.mail.handlers.text_xml"); 
     mc.addMailcap("text/plain;; x-java-content-handler=com.sun.mail.handlers.text_plain"); 
     mc.addMailcap("multipart/*;; x-java-content-handler=com.sun.mail.handlers.multipart_mixed"); 
     mc.addMailcap("message/rfc822;; x-java-content-handler=com.sun.mail.handlers.message_rfc822"); 
     CommandMap.setDefaultCommandMap(mc); 
    } 

    public Mail(String user, String pass) { 
     this(); 

     _user = user; 
     _pass = pass; 
    } 

    public boolean send() throws Exception { 
     Properties props = _setProperties(); 

     if (!_user.equals("") && !_pass.equals("") && _to.length > 0 
       && !_from.equals("") && !_subject.equals("") 
       && !_body.equals("")) { 
      Session session = Session.getInstance(props, this); 

      MimeMessage msg = new MimeMessage(session); 

      msg.setFrom(new InternetAddress(_from)); 

      InternetAddress[] addressTo = new InternetAddress[_to.length]; 
      for (int i = 0; i < _to.length; i++) { 
       addressTo[i] = new InternetAddress(_to[i]); 
      } 
      msg.setRecipients(MimeMessage.RecipientType.TO, addressTo); 

      msg.setSubject(_subject); 
      msg.setSentDate(new Date()); 

      // setup message body 
      BodyPart messageBodyPart = new MimeBodyPart(); 
      messageBodyPart.setText(_body); 
      _multipart.addBodyPart(messageBodyPart); 

      msg.setHeader("X-Priority", "1"); 
      // Put parts in message 
      msg.setContent(_multipart); 

      // send email 
      Transport.send(msg); 

      return true; 
     } else { 
      return false; 
     } 
    } 

    public void addAttachment(String filename) throws Exception { 
     BodyPart messageBodyPart = new MimeBodyPart(); 
     DataSource source = new FileDataSource(filename); 
     messageBodyPart.setDataHandler(new DataHandler(source)); 
     messageBodyPart.setFileName(filename); 

     _multipart.addBodyPart(messageBodyPart); 
    } 

    @Override 
    public PasswordAuthentication getPasswordAuthentication() { 
     return new PasswordAuthentication(_user, _pass); 
    } 

    private Properties _setProperties() { 
     Properties props = new Properties(); 

     props.put("mail.smtp.host", _host); 

     if (_debuggable) { 
      props.put("mail.debug", "true"); 
     } 

     if (_auth) { 
      props.put("mail.smtp.auth", "true"); 
     } 

     props.put("mail.smtp.port", _port); 
     props.put("mail.smtp.socketFactory.port", _sport); 
     props.put("mail.smtp.socketFactory.class", 
       "javax.net.ssl.SSLSocketFactory"); 
     props.put("mail.smtp.socketFactory.fallback", "false"); 

     return props; 
    } 

    // the getters and setters 
    public String getBody() { 
     return _body; 
    } 

    public void setBody(String _body) { 
     this._body = _body; 
    } 

    public String get_user() { 
     return _user; 
    } 

    public void set_user(String _user) { 
     this._user = _user; 
    } 

    public String get_pass() { 
     return _pass; 
    } 

    public void set_pass(String _pass) { 
     this._pass = _pass; 
    } 

    public String[] get_to() { 
     return _to; 
    } 

    public void set_to(String[] _to) { 
     this._to = _to; 
    } 

    public String get_from() { 
     return _from; 
    } 

    public void set_from(String _from) { 
     this._from = _from; 
    } 

    public String get_port() { 
     return _port; 
    } 

    public void set_port(String _port) { 
     this._port = _port; 
    } 

    public String get_sport() { 
     return _sport; 
    } 

    public void set_sport(String _sport) { 
     this._sport = _sport; 
    } 

    public String get_host() { 
     return _host; 
    } 

    public void set_host(String _host) { 
     this._host = _host; 
    } 

    public String get_subject() { 
     return _subject; 
    } 

    public void set_subject(String _subject) { 
     this._subject = _subject; 
    } 

    public boolean is_auth() { 
     return _auth; 
    } 

    public void set_auth(boolean _auth) { 
     this._auth = _auth; 
    } 

    public boolean is_debuggable() { 
     return _debuggable; 
    } 

    public void set_debuggable(boolean _debuggable) { 
     this._debuggable = _debuggable; 
    } 

    public Multipart get_multipart() { 
     return _multipart; 
    } 

    public void set_multipart(Multipart _multipart) { 
     this._multipart = _multipart; 
    } 
} 
+0

obviosuly你不能繪製文件是這樣的... 1.搶繪製對象,2.將其保存到了一些路徑,使用此路徑?如何做到既1和2在bazillions的回答類似的問題在這裏 – Selvin

+0

謝謝 - 請儘量不要粗魯。我不是專家,我正在學習。我一直在努力工作超過30個小時。這個項目是由我的老闆隨意拋給我的,他預計這個工作會在另一天完成。 – FoxDonut

+0

你可以像這樣得到可繪製路徑Uri path = Uri.parse(「android.resource://」+ BuildConfig.APPLICATION_ID +「/」+ R.drawable.image),你可以像那樣使用fileName – Kayra

回答

0

按照您的評論中的要求,我的評論作爲回答(用更多的代碼來說明問題)。

Selvin指出你正確的方向,你不能發送一個位圖作爲附件,因爲它只是和內部的數據結構,但你必須將它轉換爲收件人可以使用的東西,例如,一個PNG在你的情況。

而不是寫入文件系統,您可以將其生成內存。這樣你就不需要寫入設備存儲器的權限(如果你不需要它就可以用於其他事情),並且在成功發送郵件後你不需要清理它。

Java EE帶有一個ByteArrayDataSource,因此如果您的應用程序中具有該類,那麼您只需將ByteArrayDataSource替換爲FileDataSource,然後將壓縮後的圖像作爲字節數組返回。因爲我不知道你的項目中是否有這個類,所以接下來是一個不使用DataHandler的解決方案。

這裏是表示我的點一些代碼示例:

public byte[] getImageData(Bitmap bitmapImage) { 
    ByteArrayOutputStream bos = null; 
    try { 
     bos = new ByteArrayOutputStream(); 
     // Use the compress method on the BitMap object to write image to the OutputStream 
     bitmapImage.compress(Bitmap.CompressFormat.PNG, 100, bos); 
    } catch (Exception e) { 
     e.printStackTrace(); 
    } 
    return baos.toByteArray(); 
} 

而是返回一個字節數組數據源的。然後,將數據添加爲附件是這樣的:

public void addAttachment(byte[] data, String filename) throws Exception { 
    BodyPart messageBodyPart = new MimeBodyPart(); 
    messageBodyPart.setFileName(MimeUtility.encodeText(filename, "UTF-8", null)); 
    messageBodyPart.setContent(data, "image/png"); 
    _multipart.addBodyPart(messageBodyPart); 
} 
0

的問題是,我是試圖添加一個完全不正確的文件路徑。感謝@Selvin指出我正確的方向。

我必須將我想發送到設備內部存儲器的圖像存儲起來,然後給出該內部文件的路徑。下面是我用來將文件存儲到內部存儲器中的函數,它還返回一個String路徑名。

public String saveToInternalStorage(Bitmap bitmapImage) { 
    ContextWrapper wrapper = new ContextWrapper(getApplicationContext()); 
    File directory = wrapper.getDir("images", Context.MODE_PRIVATE); 
    File path = new File(directory, "output.png"); 

    FileOutputStream fos = null; 
    try { 
     fos = new FileOutputStream(path); 
     // Use the compress method on the BitMap object to write image to the OutputStream 
     bitmapImage.compress(Bitmap.CompressFormat.PNG, 100, fos); 
    } catch (Exception e) { 
     e.printStackTrace(); 
    } finally { 
     try { 
      fos.close(); 
     } catch (Exception e) { 
      e.printStackTrace(); 
     } 
    } 
    return path.getAbsolutePath(); 
} 

以我的功能,我用下面的代碼到我的RES /抽拉/ IMG通入上述方法:

String path = saveToInternalStorage(BitmapFactory.decodeResource(getResources(), R.drawable.add_up)); 

還應當指出的是,我有4個可繪製文件夾(drawable- ldpi,drawable-mdpi,drawable-hdpi和drawable-xhdpi)。所選的可繪製文件夾與您的設備爲其自身選擇「最佳圖像」相關。更清楚的是,具有更高分辨率的設備將從hdpi和xhdpi文件夾中獲取圖像......我想你會明白。

+0

除了FileDataSources以外,還有其他一些允許在內存中操作的方法,因此不必保存到文件系統。 – Lothar

+0

@Lothar請提供這個答案。沒有其他人能夠提供這些信息或提供比我擁有的更好的解決方案。 – FoxDonut