ওয়েবভিউতে ফাইল আপলোড


170

আমি গত কয়েক দিন থেকে ওয়েবভিউ থেকে ফাইলগুলি আপলোড করার জন্য লড়াই করে যাচ্ছি এবং কোনও অগ্রগতি নেই। আমি সমস্ত প্রস্তাবিত সমাধানগুলি গুগল করেছিলাম এবং প্রয়োগ করেছি তবে কিছুই কার্যকর হয় না, যেমন: সমাধানগুলি এখানে প্রস্তাবিত এবং আরও on

সমস্যা: আমার কাছে একটি ফাইল আপলোড করার জন্য নিম্নলিখিত কোড সহ একটি HTML পৃষ্ঠা রয়েছে। এটি ফায়ারফক্সের মতো ডেস্কটপ ব্রাউজারে এবং এমুলেটর / এভিডি এর অন্তর্নির্মিত ব্রাউজারে ঠিক কাজ করে, আমি যখন উপাদান দ্বারা রেন্ডার করা "ব্রাউজ ..." বোতামটি ক্লিক করি তখন ব্রাউজার একটি ডায়ালগ বাক্স খুলবে যেখানে আমি আপলোড করার জন্য একটি ফাইল বেছে নিতে পারি।

তবে, অ্যান্ড্রয়েড 3.0.০ এমুলেটর / এভিডি-তে, যখন আমি "ফাইল চয়ন করুন" এ ক্লিক করি তখন কিছুই হয় না, কোনও ফাইল ডায়লগ খোলা হয় না !!!

<form method="POST" enctype="multipart/form-data">
File to upload: <input type="file" name="uploadfile">&nbsp;&nbsp;
<input type="submit" value="Press to Upload..."> to upload the file!
</form>

যে কেউ দয়া করে শীঘ্রই একটি সম্ভাব্য সমাধানের পরামর্শ দিতে পারেন।


আপনি এটি ব্যবহার করতে পারে Webview: উপশ্রেণী যা ফাইল আপলোড ইত্যাদি স্বয়ংক্রিয়ভাবে পরিচালনা github.com/delight-im/Android-AdvancedWebView
কাকের ডাক

@MarcoW। আমি অ্যাডভান্সডউইভ ভিউ চেষ্টা করেছি, তবে এখনও ফাইল আপলোড করতে পারি না।
জিয়াশি

@ জিয়াশি ফাইল আপলোডগুলি অ্যান্ড্রয়েড ৪.৪ ব্যতীত সমস্ত অ্যান্ড্রয়েড সংস্করণে কাজ করে, যেখানে এটি কাজ করার কোনও সুযোগ নেই। আপনি কি এই সংস্করণে আছেন? অন্যথায়, AdvancedWebView.isFileUploadAvailable()আপলোডগুলি সমর্থন করে কিনা তা পরীক্ষা করে দেখতে পারেন। : এবং আপনি আপনার কোড ভাগ এবং বিষয় সাহায্যের জন্য অনুরোধ করতে পারেন github.com/delight-im/Android-AdvancedWebView/issues
কাকের ডাক

@ হীরাম কি আমি আমার অ্যান্ড্রয়েড অ্যাপ্লিকেশনটিতে ওয়েব ভিউতে আইফ্রেমের মাধ্যমে একটি ফাইল আপলোড করতে পারি? আমার কী করা দরকার?
গৌরব পরশার

উত্তর:


162

এটি সমস্ত অ্যান্ড্রয়েড সংস্করণগুলির জন্য একটি সম্পূর্ণ সমাধান, এটির সাথে আমারও খুব কষ্ট হয়েছিল।

public class MyWb extends Activity {
/** Called when the activity is first created. */

WebView web;
ProgressBar progressBar;

private ValueCallback<Uri> mUploadMessage;  
 private final static int FILECHOOSER_RESULTCODE=1;  

 @Override  
 protected void onActivityResult(int requestCode, int resultCode,  
                                    Intent intent) {  
  if(requestCode==FILECHOOSER_RESULTCODE)  
  {  
   if (null == mUploadMessage) return;  
            Uri result = intent == null || resultCode != RESULT_OK ? null  
                    : intent.getData();  
            mUploadMessage.onReceiveValue(result);  
            mUploadMessage = null;  
  }
  }  

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);

    web = (WebView) findViewById(R.id.webview01);
    progressBar = (ProgressBar) findViewById(R.id.progressBar1);

    web = new WebView(this);  
    web.getSettings().setJavaScriptEnabled(true);
    web.loadUrl("http://www.script-tutorials.com/demos/199/index.html");
    web.setWebViewClient(new myWebClient());
    web.setWebChromeClient(new WebChromeClient()  
    {  
           //The undocumented magic method override  
           //Eclipse will swear at you if you try to put @Override here  
        // For Android 3.0+
        public void openFileChooser(ValueCallback<Uri> uploadMsg) {  

            mUploadMessage = uploadMsg;  
            Intent i = new Intent(Intent.ACTION_GET_CONTENT);  
            i.addCategory(Intent.CATEGORY_OPENABLE);  
            i.setType("image/*");  
            MyWb.this.startActivityForResult(Intent.createChooser(i,"File Chooser"), FILECHOOSER_RESULTCODE);  

           }

        // For Android 3.0+
           public void openFileChooser( ValueCallback uploadMsg, String acceptType ) {
           mUploadMessage = uploadMsg;
           Intent i = new Intent(Intent.ACTION_GET_CONTENT);
           i.addCategory(Intent.CATEGORY_OPENABLE);
           i.setType("*/*");
           MyWb.this.startActivityForResult(
           Intent.createChooser(i, "File Browser"),
           FILECHOOSER_RESULTCODE);
           }

        //For Android 4.1
           public void openFileChooser(ValueCallback<Uri> uploadMsg, String acceptType, String capture){
               mUploadMessage = uploadMsg;  
               Intent i = new Intent(Intent.ACTION_GET_CONTENT);  
               i.addCategory(Intent.CATEGORY_OPENABLE);  
               i.setType("image/*");  
               MyWb.this.startActivityForResult( Intent.createChooser( i, "File Chooser" ), MyWb.FILECHOOSER_RESULTCODE );

           }

    });  


    setContentView(web);  


}

public class myWebClient extends WebViewClient
{
    @Override
    public void onPageStarted(WebView view, String url, Bitmap favicon) {
        // TODO Auto-generated method stub
        super.onPageStarted(view, url, favicon);
    }

    @Override
    public boolean shouldOverrideUrlLoading(WebView view, String url) {
        // TODO Auto-generated method stub

        view.loadUrl(url);
        return true;

    }

    @Override
    public void onPageFinished(WebView view, String url) {
        // TODO Auto-generated method stub
        super.onPageFinished(view, url);

        progressBar.setVisibility(View.GONE);
    }
}

//flipscreen not loading again
@Override
public void onConfigurationChanged(Configuration newConfig){        
    super.onConfigurationChanged(newConfig);
}

// To handle "Back" key press event for WebView to go back to previous screen.
/*@Override
public boolean onKeyDown(int keyCode, KeyEvent event) 
{
    if ((keyCode == KeyEvent.KEYCODE_BACK) && web.canGoBack()) {
        web.goBack();
        return true;
    }
    return super.onKeyDown(keyCode, event);
}*/
}

এছাড়াও আমি যুক্ত করতে চাই যে এই উদাহরণটির মতো "আপলোড পৃষ্ঠাটি" <4 সংস্করণে কাজ করবে না, কারণ এটিতে একটি চিত্র পূর্বরূপ বৈশিষ্ট্য রয়েছে, আপনি যদি এটি প্রাক্কলন ছাড়াই একটি সাধারণ পিএইচপি আপলোড ব্যবহার করতে চান।

আপডেট :

ললিপপ ডিভাইসগুলির জন্য সমাধানটি এখানে সন্ধান করুন এবং গন্টফেসের জন্য ধন্যবাদ

আপডেট 2 :

অ্যারিয়ো পর্যন্ত সমস্ত অ্যান্ড্রয়েড ডিভাইসগুলির সম্পূর্ণ সমাধান এখানে এবং এটি আরও উন্নত সংস্করণ , আপনার এটি দেখতে হবে, সম্ভবত এটি সহায়তা করতে পারে।


আমার সময় বাঁচানোর জন্য আপনাকে অনেক ধন্যবাদ!
রমেশ আকুলা

20
@ হাইফারার আপনি জীবনকর্মী এটি দুর্দান্তভাবে কাজ করেছে। প্লাস, আমাদের জন্য নুবস, কমপ্লিট কোড প্লেসমেন্টটি বুঝতে এবং এটি কীভাবে কাজ করে তা কেবল একগুচ্ছ ফাংশনগুলিকে পেস্ট করার পরিবর্তে বুঝতে সহায়তা করে এবং তারপর সেগুলি কোথায় রাখা এবং ব্যবহার করব তা আমরা জানি না। আপনার বিস্তারিত প্রতিক্রিয়া জন্য আপনাকে ধন্যবাদ।
পানামা জ্যাক

অনড় হয়ে যাওয়ার পরে এই পদ্ধতিটি কীভাবে পরিচালনা করবেন? আমি অবহেলা ছাড়াই একই পদ্ধতি ব্যবহার করছি এটি ভাল কাজ করে তবে যখন আমার অ্যাপ্লিকেশনটি অবলম্বনের সাথে স্থাপন করা হয় তখন কখনই এই পদ্ধতিগুলি বলা হয় না। আমি এই পদ্ধতিটি দেখেছি কেউ এই সমস্যাটি দেখে?
হর্ষবর্ধন

1
এটি কাজ করে, তবে মানটি যেমন হওয়া উচিত তেমনি আপডেট হয় না কী করে? এটি এক্সিলিপ অ্যান্ড্রয়েড API17 এমুলেটর সম্পর্কে আমার পরীক্ষা।
জেফ্রি নিও

1
progressBarসেখানে অব্যবহৃত বলে মনে হয়। আমি কেবল এটির লুকানো দেখতে পাচ্ছি, তবে এটি সক্রিয় করা হয়নি। আমি মনে করি এটির জন্য onProgressChanged(...)আপনাকে কাস্টম WebChromeClientবৈকল্পিক যুক্ত করতে হবে। (এবং ভালো কিছু getWindow().setFeatureInt( Window.FEATURE_PROGRESS, Window.PROGRESS_VISIBILITY_ON);মধ্যে onCreate।)
SZ।

106

HONEYCOMB (API 11) থেকে ওরিওতে (এপিআই 27) কাজ করার পদ্ধতি
[পাই 9.0 তে পরীক্ষিত নয়]

static WebView mWebView;
private ValueCallback<Uri> mUploadMessage;
public ValueCallback<Uri[]> uploadMessage;
public static final int REQUEST_SELECT_FILE = 100;
private final static int FILECHOOSER_RESULTCODE = 1;

সংশোধিত onActivityResult()

@Override
public void onActivityResult(int requestCode, int resultCode, Intent intent)
{
    if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP)
    {
        if (requestCode == REQUEST_SELECT_FILE)
        {
            if (uploadMessage == null)
                return;
            uploadMessage.onReceiveValue(WebChromeClient.FileChooserParams.parseResult(resultCode, intent));
            uploadMessage = null;
        }
    }
    else if (requestCode == FILECHOOSER_RESULTCODE)
    {
        if (null == mUploadMessage)
            return;
    // Use MainActivity.RESULT_OK if you're implementing WebView inside Fragment
    // Use RESULT_OK only if you're implementing WebView inside an Activity
        Uri result = intent == null || resultCode != MainActivity.RESULT_OK ? null : intent.getData();
        mUploadMessage.onReceiveValue(result);
        mUploadMessage = null;
    }
    else
        Toast.makeText(getActivity().getApplicationContext(), "Failed to Upload Image", Toast.LENGTH_LONG).show();
}

এখন onCreate()বা onCreateView()নীচের কোডটি পেস্ট করুন

    WebSettings mWebSettings = mWebView.getSettings();
    mWebSettings.setJavaScriptEnabled(true);
    mWebSettings.setSupportZoom(false);
    mWebSettings.setAllowFileAccess(true);
    mWebSettings.setAllowFileAccess(true);
    mWebSettings.setAllowContentAccess(true);

mWebView.setWebChromeClient(new WebChromeClient()
{
    // For 3.0+ Devices (Start)
    // onActivityResult attached before constructor
    protected void openFileChooser(ValueCallback uploadMsg, String acceptType)
    {
        mUploadMessage = uploadMsg;
        Intent i = new Intent(Intent.ACTION_GET_CONTENT);
        i.addCategory(Intent.CATEGORY_OPENABLE);
        i.setType("image/*");
        startActivityForResult(Intent.createChooser(i, "File Browser"), FILECHOOSER_RESULTCODE);
    }


    // For Lollipop 5.0+ Devices
    public boolean onShowFileChooser(WebView mWebView, ValueCallback<Uri[]> filePathCallback, WebChromeClient.FileChooserParams fileChooserParams)
    {
        if (uploadMessage != null) {
            uploadMessage.onReceiveValue(null);
            uploadMessage = null;
        }

        uploadMessage = filePathCallback;

        Intent intent = fileChooserParams.createIntent();
        try
        {
            startActivityForResult(intent, REQUEST_SELECT_FILE);
        } catch (ActivityNotFoundException e)
        {
            uploadMessage = null;
            Toast.makeText(getActivity().getApplicationContext(), "Cannot Open File Chooser", Toast.LENGTH_LONG).show();
            return false;
        }
        return true;
    }

    //For Android 4.1 only
    protected void openFileChooser(ValueCallback<Uri> uploadMsg, String acceptType, String capture)
    {
        mUploadMessage = uploadMsg;
        Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
        intent.addCategory(Intent.CATEGORY_OPENABLE);
        intent.setType("image/*");
        startActivityForResult(Intent.createChooser(intent, "File Browser"), FILECHOOSER_RESULTCODE);
    }

    protected void openFileChooser(ValueCallback<Uri> uploadMsg)
    {
        mUploadMessage = uploadMsg;
        Intent i = new Intent(Intent.ACTION_GET_CONTENT);
        i.addCategory(Intent.CATEGORY_OPENABLE);
        i.setType("image/*");
        startActivityForResult(Intent.createChooser(i, "File Chooser"), FILECHOOSER_RESULTCODE);
    }
});

আমার সমস্যা সমাধান। ধন্যবাদ.
Htin অং

1
আপনার সম্পূর্ণ কোডের জন্য ধন্যবাদ। কিটকাট ওস ছাড়া এটির কাজ ঠিক আছে । আপনি কি আমাকে কীটকেটে সমর্থন করবেন সে সম্পর্কে কোনও ধারণা সরবরাহ করবেন। ধন্যবাদ
অ্যান্ডিবয়

নিখুঁত উত্তর !! আপনাকে অনেক ধন্যবাদ
@


1
@ নাইটফিউরি আপনি এইটি করতে হবে, যদি আপনি কোনও HTTP লিঙ্কটি অ্যাক্সেস করার চেষ্টা করছেন । ব্যবহারের https পরিবর্তে
zackygaurav

45

এই একমাত্র সমাধান আমি খুঁজে পেয়েছি যে কাজ করে!

WebView webview;

private ValueCallback<Uri> mUploadMessage;
private final static int FILECHOOSER_RESULTCODE = 1;

@Override
protected void onActivityResult(int requestCode, int resultCode,
        Intent intent) {
    if (requestCode == FILECHOOSER_RESULTCODE) {
        if (null == mUploadMessage)
            return;
        Uri result = intent == null || resultCode != RESULT_OK ? null
                : intent.getData();
        mUploadMessage.onReceiveValue(result);
        mUploadMessage = null;

    }
}

// Next part 

class MyWebChromeClient extends WebChromeClient {
    // The undocumented magic method override
    // Eclipse will swear at you if you try to put @Override here
    public void openFileChooser(ValueCallback<Uri> uploadMsg) {

        mUploadMessage = uploadMsg;
        Intent i = new Intent(Intent.ACTION_GET_CONTENT);
        i.addCategory(Intent.CATEGORY_OPENABLE);
        i.setType("image/*");
        Cv5appActivity.this.startActivityForResult(
                Intent.createChooser(i, "Image Browser"),
                FILECHOOSER_RESULTCODE);
    }
}

3
আপনি কি আমাকে স্পষ্ট করে বলতে পারেন এই কোডটি আমার কোথায় রাখা উচিত? এবং আমি এটি কীভাবে ব্যবহার করব?
ক্যাম্পোব

27
৪.৪ (কিটকাট) হিসাবে, ওপেনফাইলেচুসার আর উপলভ্য নয়, সুতরাং এটি ব্যবহার করে এমন পদ্ধতির কাজ হয় না।
স্টিভ এন

4
@nadeemgc আমার ক্ষেত্রে, আমি ওয়েব সার্ভারটিও নিয়ন্ত্রণ করি, তাই এই ইউআরএলগুলি ব্যতীত আপলোড লিঙ্কটির জন্য একটি ভিন্ন ইউআরএল স্কিম (ওরফে "আমার-অ্যাপ: // কিছু / অন্যান্য / যুক্তি") ব্যবহার করে আমি অ্যান্ড্রয়েড 4.4+ সনাক্ত করছি am ওভাররিডউরললোডিং () এ, এবং তারপরে কোনও পিকার দেখানো এবং সেই ক্ষেত্রে স্থানীয় কোডে আপলোড করা doing এটি একেবারে হালকা ওজনের কাজ নয়।
স্টিভ এন

2
ওয়েব ভিউ ব্যবহার করে আমি কি অ্যান্ড্রয়েড ৪.৪.২ এ কোনও চিত্র আপলোড করতে পারি?
বাবু কে

@ স্টিভেন, আপনি কি একটি স্নিপেট সরবরাহ করতে পারেন? এটা অসাধারণ হবে !
জিপি 2 এমভি 3

28

৫.০-তে ললিপপ, গুগল একটি অফিসিয়াল পদ্ধতি, ওয়েবক্রোমক্লিয়েন্ট.অনশো ফাইল চুসর যুক্ত করেছে । এমনকি তারা ফাইল চয়নকারী অভিপ্রায়টি স্বয়ংক্রিয়ভাবে উত্পন্ন করার জন্য একটি উপায় সরবরাহ করে যাতে এটি ইনপুট মিম প্রকারগুলি গ্রহণ করে uses

public class MyWebChromeClient extends WebChromeClient {
        // reference to activity instance. May be unnecessary if your web chrome client is member class.
    private MyActivity activity;

    public boolean onShowFileChooser(WebView webView, ValueCallback<Uri[]> filePathCallback, FileChooserParams fileChooserParams) {
        // make sure there is no existing message
        if (myActivity.uploadMessage != null) {
            myActivity.uploadMessage.onReceiveValue(null);
            myActivity.uploadMessage = null;
        }

        myActivity.uploadMessage = filePathCallback;

        Intent intent = fileChooserParams.createIntent();
        try {
            myActivity.startActivityForResult(intent, MyActivity.REQUEST_SELECT_FILE);
        } catch (ActivityNotFoundException e) {
            myActivity.uploadMessage = null;
            Toast.makeText(myActivity, "Cannot open file chooser", Toast.LENGTH_LONG).show();
            return false;
        }

        return true;
    }
}


public class MyActivity extends ... {
    public static final int REQUEST_SELECT_FILE = 100;
    public ValueCallback<Uri[]> uploadMessage;

    protected void onActivityResult(int requestCode, int resultCode, Intent data){
        if (requestCode == REQUEST_SELECT_FILE) {
                if (uploadMessage == null) return;
                uploadMessage.onReceiveValue(WebChromeClient.FileChooserParams.parseResult(resultCode, data));
                uploadMessage = null;
            }
        }
    }
}

KitKat এর আগে অ্যান্ড্রয়েড সংস্করণগুলির জন্য, অন্যান্য উত্তরে উল্লিখিত ব্যক্তিগত পদ্ধতিগুলি কাজ করে। আমি কিটক্যাট (৪.৪) এর মতো ভাল কাজ খুঁজে পাইনি।


আমার পরিস্থিতিতে, uploadMessage.onReceiveValue()কাজ করে এবং একটি ঘনিষ্ঠতা রয়েছে)
নিনজা

এটি আমার পক্ষে কাজ করে না। আমি আপনার কোড দিয়ে 'মাই ওয়েবক্রোম ক্লিনেন্ট' নামে একটি নতুন ক্লাস তৈরি করার চেষ্টা করেছি, কিন্তু আমি পেয়েছি যে মাইএকটিভিটি এমন একটি শ্রেণি যা আমি আমদানি করতে পারি না ... আমি 'মাইএকটিভিটি' নামে নিজের একটি ক্লাস তৈরি করতে পারতাম তবে আমি কী লিখব তা জানতাম না এটার ভিতরে.
rikkitikkitumbo

19

আমি দেখতে পেয়েছি যে অ্যান্ড্রয়েডের বিভিন্ন সংস্করণ পরিচালনা করতে আমার 3 টি ইন্টারফেস সংজ্ঞা প্রয়োজন।

public void openFileChooser(ValueCallback < Uri > uploadMsg) {
  mUploadMessage = uploadMsg;
  Intent i = new Intent(Intent.ACTION_GET_CONTENT);
  i.addCategory(Intent.CATEGORY_OPENABLE);
  i.setType("image/*");
  FreeHealthTrack.this.startActivityForResult(Intent.createChooser(i, "Image Chooser"), FILECHOOSER_RESULTCODE);
}

public void openFileChooser(ValueCallback < Uri > uploadMsg, String acceptType) {
  openFileChooser(uploadMsg);
}

public void openFileChooser(ValueCallback < Uri > uploadMsg, String acceptType, String capture) {
  openFileChooser(uploadMsg);
}

3
ধন্যবাদ, জেলি বিনের জন্য শেষটি আবশ্যক, যা সে আমার পক্ষে কাজ করে না।
শালাফি

3
অনড় হয়ে যাওয়ার পরে এই পদ্ধতিটি কীভাবে পরিচালনা করবেন? আমি অবহেলা ছাড়াই একই পদ্ধতি ব্যবহার করছি এটি ভাল কাজ করে তবে যখন আমার অ্যাপ্লিকেশনটি অবলম্বনের সাথে স্থাপন করা হয় তখন কখনই এই পদ্ধতিগুলি বলা হয় না। আমি এই পদ্ধতিটি দেখেছি কেউ এই সমস্যাটি দেখে?
হর্ষবর্ধন

2
এটি অন্য উপায়ে করা আরও ভাল হবে, অর্থাৎ সর্বশেষতম একের মধ্যে কোডটি রাখুন যার মধ্যে সবচেয়ে ধনী স্বাক্ষর রয়েছে এবং প্রবর্তন করার জন্য কোন ধরণের অভিপ্রায়টি বেছে নিতে তার পরামিতিগুলি ব্যবহার করুন। তারপরে এই দুটি পরামিতিগুলির জন্য ডিফল্ট মান সহ শেষ দুটি কল করুন।
ম্যাটটিও

16

এই সমাধানটি মধুচক্র এবং আইসক্রিম স্যান্ডউইচের জন্যও কাজ করে। দেখে মনে হচ্ছে গুগল একটি দুর্দান্ত নতুন বৈশিষ্ট্য প্রবর্তন করেছে (গুণাবলী গ্রহণ করুন) এবং পিছনের সামঞ্জস্যের জন্য একটি ওভারলোড বাস্তবায়ন করতে ভুলে গেছেন।

protected class CustomWebChromeClient extends WebChromeClient
{
    // For Android 3.0+
    public void openFileChooser( ValueCallback<Uri> uploadMsg, String acceptType ) 
    {  
        context.mUploadMessage = uploadMsg;  
        Intent i = new Intent(Intent.ACTION_GET_CONTENT);  
        i.addCategory(Intent.CATEGORY_OPENABLE);  
        i.setType("image/*");  
        context.startActivityForResult( Intent.createChooser( i, "File Chooser" ), MainActivity.FILECHOOSER_RESULTCODE );  
    }

    // For Android < 3.0
    public void openFileChooser( ValueCallback<Uri> uploadMsg ) 
    {
        openFileChooser( uploadMsg, "" );
    }
}

আপনি এই ক্লাসটি কোথায় রাখা উচিত দয়া করে পরিষ্কার করতে পারেন? এবং আমি এটি কীভাবে ব্যবহার করব? ধন্যবাদ
ক্যাম্পব্যাপ

এই শ্রেণিটিকে আপনার ক্রিয়াকলাপের যে কোনও জায়গায় রাখুন এবং এই কোডটি ওয়েবভিউ.সেটউইব ক্রোমক্লিনেন্ট (নতুন কাস্টমওয়েব ক্রোমক্লায়েন্ট ()) ব্যবহার করুন;
বিশাল পওয়ার


5
৪.৪ (কিটকাট) হিসাবে, ওপেনফাইলেচুসার আর উপলভ্য নয়, সুতরাং এটি ব্যবহার করে এমন পদ্ধতির কাজ হয় না।
স্টিভ এন

16

হাইফারারের সম্পূর্ণ সমাধানটি আমার পক্ষে খুব সহায়ক।

তবে, আমি আরও অনেক সমস্যার মুখোমুখি হয়েছি - অন্যান্য মাইম প্রকারকে সমর্থন করা, ক্যাপচার ডিভাইসগুলির তালিকা (ক্যামেরা, ভিডিও, অডিও রেকর্ডার), অবিলম্বে ক্যাপচার ডিভাইস খোলার (উদা: <ইনপুট গ্রহণ = "চিত্র / *; ক্যাপচার">) ...

সুতরাং, আমি একটি সমাধান তৈরি করেছি যা ডিফল্ট ওয়েব ব্রাউজার অ্যাপের মতো ঠিক কাজ করে।

আমি অ্যান্ড্রয়েড -৪.৪.৩_আর ১ / এসসিআর / কম / অ্যান্ড্রয়েড / ব্রাউজার / আপলোডহ্যান্ডলিরজাভা ব্যবহার করেছি। (রূপ্ট রাউন্সলেকে ধন্যবাদ)

package org.mospi.agatenativewebview;

import java.io.File;
import java.lang.reflect.Method;
import java.net.URL;

import android.app.Activity;
import android.app.AlertDialog;
import android.content.ActivityNotFoundException;
import android.content.DialogInterface;
import android.content.Intent;
import android.net.Uri;
import android.os.Bundle;
import android.os.Environment;
import android.provider.MediaStore;
import android.view.View;
import android.webkit.JsResult;
import android.webkit.ValueCallback;
import android.webkit.WebChromeClient;
import android.webkit.WebSettings;
import android.webkit.WebView;
import android.webkit.WebSettings.PluginState;
import android.widget.Toast;

public class MainActivity extends Activity {

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        WebView webView = (WebView) findViewById(R.id.webView1);

        initWebView(webView);
        webView.loadUrl("http://google.com"); // TODO input your url

    }

    private final static Object methodInvoke(Object obj, String method, Class<?>[] parameterTypes, Object[] args) {
        try {
            Method m = obj.getClass().getMethod(method, new Class[] { boolean.class });
            m.invoke(obj, args);
        } catch (Exception e) {
            e.printStackTrace();
        }

        return null;
    }

    private void initWebView(WebView webView) {

        WebSettings settings = webView.getSettings();

        settings.setJavaScriptEnabled(true);
        settings.setAllowFileAccess(true);
        settings.setDomStorageEnabled(true);
        settings.setCacheMode(WebSettings.LOAD_NO_CACHE);
        settings.setLoadWithOverviewMode(true);
        settings.setUseWideViewPort(true);
        settings.setSupportZoom(true);
        // settings.setPluginsEnabled(true);
        methodInvoke(settings, "setPluginsEnabled", new Class[] { boolean.class }, new Object[] { true });
        // settings.setPluginState(PluginState.ON);
        methodInvoke(settings, "setPluginState", new Class[] { PluginState.class }, new Object[] { PluginState.ON });
        // settings.setPluginsEnabled(true);
        methodInvoke(settings, "setPluginsEnabled", new Class[] { boolean.class }, new Object[] { true });
        // settings.setAllowUniversalAccessFromFileURLs(true);
        methodInvoke(settings, "setAllowUniversalAccessFromFileURLs", new Class[] { boolean.class }, new Object[] { true });
        // settings.setAllowFileAccessFromFileURLs(true);
        methodInvoke(settings, "setAllowFileAccessFromFileURLs", new Class[] { boolean.class }, new Object[] { true });

        webView.setScrollBarStyle(View.SCROLLBARS_INSIDE_OVERLAY);
        webView.clearHistory();
        webView.clearFormData();
        webView.clearCache(true);

        webView.setWebChromeClient(new MyWebChromeClient());
        // webView.setDownloadListener(downloadListener);
    }

    UploadHandler mUploadHandler;

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent intent) {

        if (requestCode == Controller.FILE_SELECTED) {
            // Chose a file from the file picker.
            if (mUploadHandler != null) {
                mUploadHandler.onResult(resultCode, intent);
            }
        }

        super.onActivityResult(requestCode, resultCode, intent);
    }

    class MyWebChromeClient extends WebChromeClient {
        public MyWebChromeClient() {

        }

        private String getTitleFromUrl(String url) {
            String title = url;
            try {
                URL urlObj = new URL(url);
                String host = urlObj.getHost();
                if (host != null && !host.isEmpty()) {
                    return urlObj.getProtocol() + "://" + host;
                }
                if (url.startsWith("file:")) {
                    String fileName = urlObj.getFile();
                    if (fileName != null && !fileName.isEmpty()) {
                        return fileName;
                    }
                }
            } catch (Exception e) {
                // ignore
            }

            return title;
        }

        @Override
        public boolean onJsAlert(WebView view, String url, String message, final JsResult result) {
            String newTitle = getTitleFromUrl(url);

            new AlertDialog.Builder(MainActivity.this).setTitle(newTitle).setMessage(message).setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() {

                @Override
                public void onClick(DialogInterface dialog, int which) {
                    result.confirm();
                }
            }).setCancelable(false).create().show();
            return true;
            // return super.onJsAlert(view, url, message, result);
        }

        @Override
        public boolean onJsConfirm(WebView view, String url, String message, final JsResult result) {

            String newTitle = getTitleFromUrl(url);

            new AlertDialog.Builder(MainActivity.this).setTitle(newTitle).setMessage(message).setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() {

                @Override
                public void onClick(DialogInterface dialog, int which) {
                    result.confirm();
                }
            }).setNegativeButton(android.R.string.cancel, new DialogInterface.OnClickListener() {
                public void onClick(DialogInterface dialog, int which) {
                    result.cancel();
                }
            }).setCancelable(false).create().show();
            return true;

            // return super.onJsConfirm(view, url, message, result);
        }

        // Android 2.x
        public void openFileChooser(ValueCallback<Uri> uploadMsg) {
            openFileChooser(uploadMsg, "");
        }

        // Android 3.0
        public void openFileChooser(ValueCallback<Uri> uploadMsg, String acceptType) {
            openFileChooser(uploadMsg, "", "filesystem");
        }

        // Android 4.1
        public void openFileChooser(ValueCallback<Uri> uploadMsg, String acceptType, String capture) {
            mUploadHandler = new UploadHandler(new Controller());
            mUploadHandler.openFileChooser(uploadMsg, acceptType, capture);
        }

        // Android 4.4, 4.4.1, 4.4.2
        // openFileChooser function is not called on Android 4.4, 4.4.1, 4.4.2,
        // you may use your own java script interface or other hybrid framework.          

        // Android 5.0.1
        public boolean onShowFileChooser(
                WebView webView, ValueCallback<Uri[]> filePathCallback,
                FileChooserParams fileChooserParams) {

            String acceptTypes[] = fileChooserParams.getAcceptTypes();

            String acceptType = "";
            for (int i = 0; i < acceptTypes.length; ++ i) {
                if (acceptTypes[i] != null && acceptTypes[i].length() != 0)
                    acceptType += acceptTypes[i] + ";";
            }
            if (acceptType.length() == 0)
                acceptType = "*/*";

            final ValueCallback<Uri[]> finalFilePathCallback = filePathCallback;

            ValueCallback<Uri> vc = new ValueCallback<Uri>() {

                @Override
                public void onReceiveValue(Uri value) {

                    Uri[] result;
                    if (value != null)
                        result = new Uri[]{value};
                    else
                        result = null;

                    finalFilePathCallback.onReceiveValue(result);

                }
            };

            openFileChooser(vc, acceptType, "filesystem");


            return true;
       }
    };

    class Controller {
        final static int FILE_SELECTED = 4;

        Activity getActivity() {
            return MainActivity.this;
        }
    }

    // copied from android-4.4.3_r1/src/com/android/browser/UploadHandler.java
    //////////////////////////////////////////////////////////////////////

    /*
     * Copyright (C) 2010 The Android Open Source Project
     *
     * Licensed under the Apache License, Version 2.0 (the "License");
     * you may not use this file except in compliance with the License.
     * You may obtain a copy of the License at
     *
     *      http://www.apache.org/licenses/LICENSE-2.0
     *
     * Unless required by applicable law or agreed to in writing, software
     * distributed under the License is distributed on an "AS IS" BASIS,
     * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     * See the License for the specific language governing permissions and
     * limitations under the License.
     */

    // package com.android.browser;
    //
    // import android.app.Activity;
    // import android.content.ActivityNotFoundException;
    // import android.content.Intent;
    // import android.net.Uri;
    // import android.os.Environment;
    // import android.provider.MediaStore;
    // import android.webkit.ValueCallback;
    // import android.widget.Toast;
    //
    // import java.io.File;
    // import java.util.Vector;
    //
    // /**
    // * Handle the file upload callbacks from WebView here
    // */
    // public class UploadHandler {

    class UploadHandler {
        /*
         * The Object used to inform the WebView of the file to upload.
         */
        private ValueCallback<Uri> mUploadMessage;
        private String mCameraFilePath;
        private boolean mHandled;
        private boolean mCaughtActivityNotFoundException;
        private Controller mController;
        public UploadHandler(Controller controller) {
            mController = controller;
        }
        String getFilePath() {
            return mCameraFilePath;
        }
        boolean handled() {
            return mHandled;
        }
        void onResult(int resultCode, Intent intent) {
            if (resultCode == Activity.RESULT_CANCELED && mCaughtActivityNotFoundException) {
                // Couldn't resolve an activity, we are going to try again so skip
                // this result.
                mCaughtActivityNotFoundException = false;
                return;
            }
            Uri result = intent == null || resultCode != Activity.RESULT_OK ? null
                    : intent.getData();
            // As we ask the camera to save the result of the user taking
            // a picture, the camera application does not return anything other
            // than RESULT_OK. So we need to check whether the file we expected
            // was written to disk in the in the case that we
            // did not get an intent returned but did get a RESULT_OK. If it was,
            // we assume that this result has came back from the camera.
            if (result == null && intent == null && resultCode == Activity.RESULT_OK) {
                File cameraFile = new File(mCameraFilePath);
                if (cameraFile.exists()) {
                    result = Uri.fromFile(cameraFile);
                    // Broadcast to the media scanner that we have a new photo
                    // so it will be added into the gallery for the user.
                    mController.getActivity().sendBroadcast(
                            new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, result));
                }
            }
            mUploadMessage.onReceiveValue(result);
            mHandled = true;
            mCaughtActivityNotFoundException = false;
        }
        void openFileChooser(ValueCallback<Uri> uploadMsg, String acceptType, String capture) {
            final String imageMimeType = "image/*";
            final String videoMimeType = "video/*";
            final String audioMimeType = "audio/*";
            final String mediaSourceKey = "capture";
            final String mediaSourceValueCamera = "camera";
            final String mediaSourceValueFileSystem = "filesystem";
            final String mediaSourceValueCamcorder = "camcorder";
            final String mediaSourceValueMicrophone = "microphone";
            // According to the spec, media source can be 'filesystem' or 'camera' or 'camcorder'
            // or 'microphone' and the default value should be 'filesystem'.
            String mediaSource = mediaSourceValueFileSystem;
            if (mUploadMessage != null) {
                // Already a file picker operation in progress.
                return;
            }
            mUploadMessage = uploadMsg;
            // Parse the accept type.
            String params[] = acceptType.split(";");
            String mimeType = params[0];
            if (capture.length() > 0) {
                mediaSource = capture;
            }
            if (capture.equals(mediaSourceValueFileSystem)) {
                // To maintain backwards compatibility with the previous implementation
                // of the media capture API, if the value of the 'capture' attribute is
                // "filesystem", we should examine the accept-type for a MIME type that
                // may specify a different capture value.
                for (String p : params) {
                    String[] keyValue = p.split("=");
                    if (keyValue.length == 2) {
                        // Process key=value parameters.
                        if (mediaSourceKey.equals(keyValue[0])) {
                            mediaSource = keyValue[1];
                        }
                    }
                }
            }
            //Ensure it is not still set from a previous upload.
            mCameraFilePath = null;
            if (mimeType.equals(imageMimeType)) {
                if (mediaSource.equals(mediaSourceValueCamera)) {
                    // Specified 'image/*' and requested the camera, so go ahead and launch the
                    // camera directly.
                    startActivity(createCameraIntent());
                    return;
                } else {
                    // Specified just 'image/*', capture=filesystem, or an invalid capture parameter.
                    // In all these cases we show a traditional picker filetered on accept type
                    // so launch an intent for both the Camera and image/* OPENABLE.
                    Intent chooser = createChooserIntent(createCameraIntent());
                    chooser.putExtra(Intent.EXTRA_INTENT, createOpenableIntent(imageMimeType));
                    startActivity(chooser);
                    return;
                }
            } else if (mimeType.equals(videoMimeType)) {
                if (mediaSource.equals(mediaSourceValueCamcorder)) {
                    // Specified 'video/*' and requested the camcorder, so go ahead and launch the
                    // camcorder directly.
                    startActivity(createCamcorderIntent());
                    return;
               } else {
                    // Specified just 'video/*', capture=filesystem or an invalid capture parameter.
                    // In all these cases we show an intent for the traditional file picker, filtered
                    // on accept type so launch an intent for both camcorder and video/* OPENABLE.
                    Intent chooser = createChooserIntent(createCamcorderIntent());
                    chooser.putExtra(Intent.EXTRA_INTENT, createOpenableIntent(videoMimeType));
                    startActivity(chooser);
                    return;
                }
            } else if (mimeType.equals(audioMimeType)) {
                if (mediaSource.equals(mediaSourceValueMicrophone)) {
                    // Specified 'audio/*' and requested microphone, so go ahead and launch the sound
                    // recorder.
                    startActivity(createSoundRecorderIntent());
                    return;
                } else {
                    // Specified just 'audio/*',  capture=filesystem of an invalid capture parameter.
                    // In all these cases so go ahead and launch an intent for both the sound
                    // recorder and audio/* OPENABLE.
                    Intent chooser = createChooserIntent(createSoundRecorderIntent());
                    chooser.putExtra(Intent.EXTRA_INTENT, createOpenableIntent(audioMimeType));
                    startActivity(chooser);
                    return;
                }
            }
            // No special handling based on the accept type was necessary, so trigger the default
            // file upload chooser.
            startActivity(createDefaultOpenableIntent());
        }
        private void startActivity(Intent intent) {
            try {
                mController.getActivity().startActivityForResult(intent, Controller.FILE_SELECTED);
            } catch (ActivityNotFoundException e) {
                // No installed app was able to handle the intent that
                // we sent, so fallback to the default file upload control.
                try {
                    mCaughtActivityNotFoundException = true;
                    mController.getActivity().startActivityForResult(createDefaultOpenableIntent(),
                            Controller.FILE_SELECTED);
                } catch (ActivityNotFoundException e2) {
                    // Nothing can return us a file, so file upload is effectively disabled.
                    Toast.makeText(mController.getActivity(), R.string.uploads_disabled,
                            Toast.LENGTH_LONG).show();
                }
            }
        }
        private Intent createDefaultOpenableIntent() {
            // Create and return a chooser with the default OPENABLE
            // actions including the camera, camcorder and sound
            // recorder where available.
            Intent i = new Intent(Intent.ACTION_GET_CONTENT);
            i.addCategory(Intent.CATEGORY_OPENABLE);
            i.setType("*/*");
            Intent chooser = createChooserIntent(createCameraIntent(), createCamcorderIntent(),
                    createSoundRecorderIntent());
            chooser.putExtra(Intent.EXTRA_INTENT, i);
            return chooser;
        }
        private Intent createChooserIntent(Intent... intents) {
            Intent chooser = new Intent(Intent.ACTION_CHOOSER);
            chooser.putExtra(Intent.EXTRA_INITIAL_INTENTS, intents);
            chooser.putExtra(Intent.EXTRA_TITLE,
                    mController.getActivity().getResources()
                            .getString(R.string.choose_upload));
            return chooser;
        }
        private Intent createOpenableIntent(String type) {
            Intent i = new Intent(Intent.ACTION_GET_CONTENT);
            i.addCategory(Intent.CATEGORY_OPENABLE);
            i.setType(type);
            return i;
        }
        private Intent createCameraIntent() {
            Intent cameraIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
            File externalDataDir = Environment.getExternalStoragePublicDirectory(
                    Environment.DIRECTORY_DCIM);
            File cameraDataDir = new File(externalDataDir.getAbsolutePath() +
                    File.separator + "browser-photos");
            cameraDataDir.mkdirs();
            mCameraFilePath = cameraDataDir.getAbsolutePath() + File.separator +
                    System.currentTimeMillis() + ".jpg";
            cameraIntent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(new File(mCameraFilePath)));
            return cameraIntent;
        }
        private Intent createCamcorderIntent() {
            return new Intent(MediaStore.ACTION_VIDEO_CAPTURE);
        }
        private Intent createSoundRecorderIntent() {
            return new Intent(MediaStore.Audio.Media.RECORD_SOUND_ACTION);
        }
    }
}

রেস / মান / স্ট্রিং.এক্সএমএল অতিরিক্ত স্ট্রিং রেজোরাস:

<string name="uploads_disabled">File uploads are disabled.</string>
<string name="choose_upload">Choose file for upload</string>

আপনি যদি অগ্রগতি ব্যবহার করে থাকেন তবে আপনার প্রগর্ড-প্রজেক্ট.টেক্সটে নীচের বিকল্পের প্রয়োজন হতে পারে:

-keepclassmembers class * extends android.webkit.WebChromeClient {
   public void openFileChooser(...);
}

আপডেট # 1 (2015.09.09)

অ্যান্ড্রয়েড 5.0.1 সংযোগের জন্য কোড যুক্ত করে।


৪.৪.২ এ এটি কাজ করতে পারবেন না। আপনি কি এটিকে অ্যান্ড্রয়েড ৪.৪.২ এ পরীক্ষা করেছেন
আনা ল্লেরা

2
অ্যানড্রয়েড ৪.৪, ৪.৪.১, ৪.৪.২ এ অন্নাদে ওপেনফাইলেচুজার ফাংশনটি কল করা হয়নি এই বাগটি অ্যান্ড্রয়েড ৪.৪.৩ এ স্থির করা হয়েছিল। রেফ।) http://code.google.com/p/android/issues/detail?id=62220 তবে, দুঃখের বিষয়, অনেক লোক 4.4.2 ব্যবহার করে। আমি মনে করি আপনি নিজের জাভা স্ক্রিপ্ট ইন্টারফেস ব্যবহার করতে পারেন (ড্যানেলকের উত্তর দেখুন) বা অন্যান্য সংকর কাঠামো (কর্ডোভা, ফোনগ্যাপ, অ্যাগেট ...)। আমার ক্ষেত্রে, আমি সার্ভারে এইচটিএমএল ফাইলগুলি সংশোধন করতে পারি না। সুতরাং আমি অ্যাগেট ওয়েবভিউ জাভা স্ক্রিপ্ট প্লাগইন ব্যবহার করেছি। অগায়েটভিউভিউ
ফাইল

@ অজ্ঞাত স্ট্যাক আমি অ্যাজেট ওয়েবভিউ ফিল আপলোড প্রয়োগ করার চেষ্টা করছি। এইচটিএম পৃষ্ঠার পরিবর্তে আমি ওয়েব ভিউতে একটি ইউআরএল সেট করতে চাই ,. আমি কেমন করে ঐটি করি?
তুষার নারাং

@ তুষারনারং "সম্পদ / মোমেল / ইউআই / ওয়েবভিউ.এক্সএমএল" ফাইলটি খুলুন। তারপরে, AGATEWEBVIEW.src বৈশিষ্ট্যটি অনুসন্ধান করুন এবং আপনার URL এ "index.htm" মানটি প্রতিস্থাপন করুন। (রেফার। github.com/applusform/WebViewFileUploadFix )
অজানা

@UnknownStack। যদি আমি পিছনে বোতাম টিপুন তবে অ্যাপ্লিকেশনটি বন্ধ হয়ে যাবে।যেখানে আপনার কোডে এই টুকরা কোডটি ব্যবহার করতে পারেন ..পাবলিক বুলিয়ান অনকায়ডাউন (ইন কী কীড, কী-ইভেন্ট ইভেন্ট) {যদি ((কীকোড == কীআভেস্ট.কি.ই.সি.সি.ডি.বি.এ.সি.কি)) এবং & এই.উইবভিউ.কনজিওব্যাক ( )) {this.webView.goBack (); সত্য প্রত্যাবর্তন; super সুপার.অনেকেডাউন (কী কোড, ইভেন্ট) ফিরিয়ে দিন; }
sunil y

12

এটা আমার জন্য কাজ। নওগাত এবং মার্শমেলোর জন্যও কাজ করুন [ 2][3]

import android.Manifest;
import android.annotation.SuppressLint;
import android.app.Activity;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.content.res.Configuration;
import android.net.Uri;
import android.os.Build;
import android.os.Bundle;
import android.os.Environment;
import android.provider.MediaStore;
import android.support.annotation.NonNull;
import android.support.v4.app.ActivityCompat;
import android.support.v4.content.ContextCompat;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.KeyEvent;
import android.view.View;
import android.webkit.ValueCallback;
import android.webkit.WebChromeClient;
import android.webkit.WebSettings;
import android.webkit.WebView;
import android.webkit.WebViewClient;
import android.widget.Toast;

import java.io.File;
import java.io.IOException;
import java.text.SimpleDateFormat;
import java.util.Date;

public class MainActivity extends AppCompatActivity {
    private static final String TAG = MainActivity.class.getSimpleName();
    private final static int FCR = 1;
    WebView webView;
    private String mCM;
    private ValueCallback<Uri> mUM;
    private ValueCallback<Uri[]> mUMA;

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent intent) {
        super.onActivityResult(requestCode, resultCode, intent);

        if (Build.VERSION.SDK_INT >= 21) {
            Uri[] results = null;

            //Check if response is positive
            if (resultCode == Activity.RESULT_OK) {
                if (requestCode == FCR) {

                    if (null == mUMA) {
                        return;
                    }
                    if (intent == null) {
                        //Capture Photo if no image available
                        if (mCM != null) {
                            results = new Uri[]{Uri.parse(mCM)};
                        }
                    } else {
                        String dataString = intent.getDataString();
                        if (dataString != null) {
                            results = new Uri[]{Uri.parse(dataString)};
                        }
                    }
                }
            }
            mUMA.onReceiveValue(results);
            mUMA = null;
        } else {

            if (requestCode == FCR) {
                if (null == mUM) return;
                Uri result = intent == null || resultCode != RESULT_OK ? null : intent.getData();
                mUM.onReceiveValue(result);
                mUM = null;
            }
        }
    }

    @SuppressLint({"SetJavaScriptEnabled", "WrongViewCast"})
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        if (Build.VERSION.SDK_INT >= 23 && (ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED || ContextCompat.checkSelfPermission(this, Manifest.permission.CAMERA) != PackageManager.PERMISSION_GRANTED)) {
            ActivityCompat.requestPermissions(MainActivity.this, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE, Manifest.permission.CAMERA}, 1);
        }

        webView = (WebView) findViewById(R.id.ifView);
        assert webView != null;

        WebSettings webSettings = webView.getSettings();
        webSettings.setJavaScriptEnabled(true);
        webSettings.setAllowFileAccess(true);

        if (Build.VERSION.SDK_INT >= 21) {
            webSettings.setMixedContentMode(0);
            webView.setLayerType(View.LAYER_TYPE_HARDWARE, null);
        } else if (Build.VERSION.SDK_INT >= 19) {
            webView.setLayerType(View.LAYER_TYPE_HARDWARE, null);
        } else if (Build.VERSION.SDK_INT < 19) {
            webView.setLayerType(View.LAYER_TYPE_SOFTWARE, null);
        }

        webView.setWebViewClient(new Callback());
        webView.loadUrl("https://infeeds.com/");
        webView.setWebChromeClient(new WebChromeClient() {

            //For Android 3.0+
            public void openFileChooser(ValueCallback<Uri> uploadMsg) {

                mUM = uploadMsg;
                Intent i = new Intent(Intent.ACTION_GET_CONTENT);
                i.addCategory(Intent.CATEGORY_OPENABLE);
                i.setType("*/*");
                MainActivity.this.startActivityForResult(Intent.createChooser(i, "File Chooser"), FCR);
            }

            // For Android 3.0+, above method not supported in some android 3+ versions, in such case we use this
            public void openFileChooser(ValueCallback uploadMsg, String acceptType) {

                mUM = uploadMsg;
                Intent i = new Intent(Intent.ACTION_GET_CONTENT);
                i.addCategory(Intent.CATEGORY_OPENABLE);
                i.setType("*/*");
                MainActivity.this.startActivityForResult(
                        Intent.createChooser(i, "File Browser"),
                        FCR);
            }

            //For Android 4.1+
            public void openFileChooser(ValueCallback<Uri> uploadMsg, String acceptType, String capture) {

                mUM = uploadMsg;
                Intent i = new Intent(Intent.ACTION_GET_CONTENT);
                i.addCategory(Intent.CATEGORY_OPENABLE);
                i.setType("*/*");
                MainActivity.this.startActivityForResult(Intent.createChooser(i, "File Chooser"), MainActivity.FCR);
            }

            //For Android 5.0+
            public boolean onShowFileChooser(
                    WebView webView, ValueCallback<Uri[]> filePathCallback,
                    WebChromeClient.FileChooserParams fileChooserParams) {

                if (mUMA != null) {
                    mUMA.onReceiveValue(null);
                }

                mUMA = filePathCallback;
                Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
                if (takePictureIntent.resolveActivity(MainActivity.this.getPackageManager()) != null) {

                    File photoFile = null;

                    try {
                        photoFile = createImageFile();
                        takePictureIntent.putExtra("PhotoPath", mCM);
                    } catch (IOException ex) {
                        Log.e(TAG, "Image file creation failed", ex);
                    }
                    if (photoFile != null) {
                        mCM = "file:" + photoFile.getAbsolutePath();
                        takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(photoFile));
                    } else {
                        takePictureIntent = null;
                    }
                }

                Intent contentSelectionIntent = new Intent(Intent.ACTION_GET_CONTENT);
                contentSelectionIntent.addCategory(Intent.CATEGORY_OPENABLE);
                contentSelectionIntent.setType("*/*");
                Intent[] intentArray;

                if (takePictureIntent != null) {
                    intentArray = new Intent[]{takePictureIntent};
                } else {
                    intentArray = new Intent[0];
                }

                Intent chooserIntent = new Intent(Intent.ACTION_CHOOSER);
                chooserIntent.putExtra(Intent.EXTRA_INTENT, contentSelectionIntent);
                chooserIntent.putExtra(Intent.EXTRA_TITLE, "Image Chooser");
                chooserIntent.putExtra(Intent.EXTRA_INITIAL_INTENTS, intentArray);
                startActivityForResult(chooserIntent, FCR);

                return true;
            }
        });
    }

    // Create an image file
    private File createImageFile() throws IOException {

        @SuppressLint("SimpleDateFormat") String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss").format(new Date());
        String imageFileName = "img_" + timeStamp + "_";
        File storageDir = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES);
        return File.createTempFile(imageFileName, ".jpg", storageDir);
    }

    @Override
    public boolean onKeyDown(int keyCode, @NonNull KeyEvent event) {

        if (event.getAction() == KeyEvent.ACTION_DOWN) {

            switch (keyCode) {
                case KeyEvent.KEYCODE_BACK:

                    if (webView.canGoBack()) {
                        webView.goBack();
                    } else {
                        finish();
                    }

                    return true;
            }
        }

        return super.onKeyDown(keyCode, event);
    }

    @Override
    public void onConfigurationChanged(Configuration newConfig) {
        super.onConfigurationChanged(newConfig);
    }

    public class Callback extends WebViewClient {
        public void onReceivedError(WebView view, int errorCode, String description, String failingUrl) {
            Toast.makeText(getApplicationContext(), "Failed loading app!", Toast.LENGTH_SHORT).show();
        }
    }
}

11

public void openFileChooser(ValueCallback<Uri> uploadMsg, String acceptType, String capture)অ্যান্ড্রয়েড ৪.১-এ, আমি এটি সংজ্ঞায়নের জন্য প্রয়োজনীয় বলে মনে করেছি । তারপরে আমি মিশেল অলিভিয়ারের সমাধান অনুসরণ করি।


আপনি কি আমাকে এই পদ্ধতিটি সংজ্ঞায়িত করতে হবে দয়া করে পরিষ্কার করতে পারেন? এবং আমি এটি কীভাবে ব্যবহার করব?
ক্যামব্যাপ

9

আইভ আসলে প্রকৃতপক্ষে ফাইল পিকারটি কিটকেটে প্রদর্শিত হতে পারে, একটি চিত্র নির্বাচন করতে পারে এবং ফাইলপাথটি ক্রিয়াকলাপের ফলাফলের জন্য পেয়ে যায় তবে কেবলমাত্র এই জিনিসটি "ঠিক করতে" সক্ষম হয় না (কারণ এই সমস্যাটির কারণ) ফাইলটি ইনপুট ফাইল করার জন্য তৈরি করা হয় ফাইল তথ্য দিয়ে আউট।

কেউ কীভাবে কোনও কার্যকলাপ থেকে ইনপুট-ফিল্ড অ্যাক্সেস করবেন তা কোনও উপায়ে জানেন? এই উদাহরণ মন্তব্য ব্যবহার করছি । কেবলমাত্র এই শেষ টুকরাটি, প্রাচীরের শেষ ইট যা আমাকে ঠিক জায়গায় রাখতে হবে (যাতে আমি কোড থেকে সরাসরি চিত্রের আপলোডটি ট্রিগার করতে পারি)।

আপডেট # 1

আমি কোনও অ্যান্ড্রয়েড দেব না তাই আমি নবাগত স্তরে কোডটি দেখাব। আমি ইতিমধ্যে বিদ্যমান ক্রিয়াকলাপে একটি নতুন কার্যকলাপ তৈরি করছি

প্রকাশ অংশ

<uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
<application android:label="TestApp">
 <activity android:name=".BrowseActivity"></activity>
</application>

এই উদাহরণের উত্তর থেকে আমার ব্রাউজএকটিভিটি ক্লাস তৈরি করছি । ওয়েবক্রোমক্লিয়েন্ট () উদাহরণটি পিকর ইউআই অংশটি ট্রিগার করে শেষ টুকরা বাদে মূলত একই দেখাচ্ছে ...

private final static int FILECHOOSER_RESULTCODE=1;  
private final static int KITKAT_RESULTCODE = 2;

...

// The new WebChromeClient() looks pretty much the same, except one piece...

WebChromeClient chromeClient = new WebChromeClient(){  
    // For Android 3.0+
    public void openFileChooser(ValueCallback<Uri> uploadMsg) { /* Default code */ }  

    // For Android 3.0+
    public void openFileChooser( ValueCallback uploadMsg, String acceptType ) { /* Default code */ }  

    //For Android 4.1, also default but it'll be as example
    public void openFileChooser(ValueCallback<Uri> uploadMsg, String acceptType, String capture){
        mUploadMessage = uploadMsg;
        Intent i = new Intent(Intent.ACTION_GET_CONTENT);
        i.addCategory(Intent.CATEGORY_OPENABLE);
        i.setType("*/*");
        BrowseActivity.this.startActivityForResult(Intent.createChooser(i, "File Chooser"), BrowseActivity.FILECHOOSER_RESULTCODE);

    }  

    // The new code
    public void showPicker( ValueCallback<Uri> uploadMsg ){  
        // Here is part of the issue, the uploadMsg is null since it is not triggered from Android
        mUploadMessage = uploadMsg; 
        Intent i = new Intent(Intent.ACTION_GET_CONTENT);
        i.addCategory(Intent.CATEGORY_OPENABLE);
        i.setType("*/*");
        BrowseActivity.this.startActivityForResult(Intent.createChooser(i, "File Chooser"), BrowseActivity.KITKAT_RESULTCODE);
    }}

এবং আরও কিছু স্টাফ

web = new WebView(this);
// Notice this part, setting chromeClient as js interface is just lazy
web.getSettings().setJavaScriptEnabled(true);
web.addJavascriptInterface(chromeClient, "jsi" );
web.getSettings().setAllowFileAccess(true);
web.getSettings().setAllowContentAccess(true);
web.clearCache(true);
web.loadUrl( "http://as3breeze.com/upload.html" );
web.setWebViewClient(new myWebClient());
web.setWebChromeClient(chromeClient);


@Override protected void onActivityResult(int requestCode, int resultCode, Intent intent) {  
  Log.d("Result", "("+requestCode+ ") - (" +resultCode  + ") - (" + intent + ") - " + mUploadMessage);  
    if (null == intent) return;  
    Uri result = null;  
    if(requestCode==FILECHOOSER_RESULTCODE)  
    {  
        Log.d("Result","Old android");  
        if (null == mUploadMessage) return;  
        result = intent == null || resultCode != RESULT_OK ? null  : intent.getData();  
        mUploadMessage.onReceiveValue(result);  
        mUploadMessage = null;  
    } else if (requestCode == KITKAT_RESULTCODE) {  
        Log.d("Result","Kitkat android");  
        result = intent.getData();  
        final int takeFlags = intent.getFlags() & (Intent.FLAG_GRANT_READ_URI_PERMISSION  | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);  
        String path = getPath( this, result);  
        File selectedFile = new File(path); 
//I used you example with a bit of editing so thought i would share, here i added a method to upload the file to the webserver
File selectedFile = new File(path);  
            UploadFile(selectedFile);


        //mUploadMessage.onReceiveValue( Uri.parse(selectedFile.toString()) );  
        // Now we have the file but since mUploadMessage was null, it gets errors
    }  
}

 public void UploadFile(File selectedFile)
{
    Random rnd = new Random();
    String sName = "File" + rnd.nextInt(999999) + selectedFile.getAbsolutePath().substring(selectedFile.getAbsolutePath().lastIndexOf("."));
    UploadedFileName = sName;
    uploadFile = selectedFile;
    if (progressBar != null && progressBar.isShowing())
    {
        progressBar.dismiss();
    }
 // prepare for a progress bar dialog
    progressBar = new ProgressDialog(mContext);
    progressBar.setCancelable(true);
    progressBar.setMessage("Uploading File");
    progressBar.setProgressStyle(ProgressDialog.STYLE_SPINNER);            
    progressBar.show();
    new Thread() {

        public void run() 
        {
            int serverResponseCode;
            String serverResponseMessage;
            HttpURLConnection connection = null;
            DataOutputStream outputStream = null;
            DataInputStream inputStream = null;
            String pathToOurFile = uploadFile.getAbsolutePath();
            String urlServer = "http://serveraddress/Scripts/UploadHandler.php?name" + UploadedFileName;
            String lineEnd = "\r\n";
            String twoHyphens = "--";
            String boundary =  "*****";

            int bytesRead, bytesAvailable, bufferSize;
            byte[] buffer;
            int maxBufferSize = 1*1024*1024;

            try
            {
                FileInputStream fileInputStream = new FileInputStream(uploadFile);

                URL url = new URL(urlServer);
                connection = (HttpURLConnection) url.openConnection();
                Log.i("File", urlServer);

                // Allow Inputs &amp; Outputs.
                connection.setDoInput(true);
                connection.setDoOutput(true);
                connection.setUseCaches(false);

                // Set HTTP method to POST.
                connection.setRequestMethod("POST");

                connection.setRequestProperty("Connection", "Keep-Alive");
                connection.setRequestProperty("Content-Type", "multipart/form-data;boundary="+boundary);
                Log.i("File", "Open conn");

                outputStream = new DataOutputStream( connection.getOutputStream() );

                outputStream.writeBytes(twoHyphens + boundary + lineEnd);
                outputStream.writeBytes("Content-Disposition: form-data; name=\"uploadedfile\";filename=\"" + pathToOurFile +"\"" + lineEnd);
                outputStream.writeBytes(lineEnd);
                Log.i("File", "write bytes");

                bytesAvailable = fileInputStream.available();
                bufferSize = Math.min(bytesAvailable, maxBufferSize);
                buffer = new byte[bufferSize];
                Log.i("File", "available: " + fileInputStream.available());

                // Read file
                bytesRead = fileInputStream.read(buffer, 0, bufferSize);

                Log.i("file", "Bytes Read: " + bytesRead);
                while (bytesRead > 0)
                {
                    outputStream.write(buffer, 0, bufferSize);
                    bytesAvailable = fileInputStream.available();
                    bufferSize = Math.min(bytesAvailable, maxBufferSize);
                    bytesRead = fileInputStream.read(buffer, 0, bufferSize);
                }

                outputStream.writeBytes(lineEnd);
                outputStream.writeBytes(twoHyphens + boundary + twoHyphens + lineEnd);

                // Responses from the server (code and message)
                serverResponseCode = connection.getResponseCode();
                serverResponseMessage = connection.getResponseMessage();
                Log.i("file repsonse", serverResponseMessage);

//once the file is uploaded call a javascript function to verify the user wants to save the image
                progressBar.dismiss();
                runOnUiThread(new Runnable() 
                {

                    @Override
                    public void run() 
                    {
                        Log.i("start", "File name: " + UploadedFileName);
                        WebView myWebView = (WebView) findViewById(R.id.webview);
                        myWebView.loadUrl("javascript:CheckImage('" + UploadedFileName + "')");
                    }
                });


                fileInputStream.close();
                outputStream.flush();
                outputStream.close();
            }
            catch (Exception ex)
            {
                Log.i("exception", "Error: " + ex.toString());
            }               
        }
    }.start();

}

শেষ অবধি, আসল ফাইলের পাথ পেতে আরও কিছু কোড, এসও-তে পাওয়া কোড, ive মন্তব্যগুলিতে পোস্ট ইউআরএল যুক্ত করেছে যাতে লেখক তার কাজের জন্য ক্রেডিট পান।

/**
 * Get a file path from a Uri. This will get the the path for Storage Access
 * Framework Documents, as well as the _data field for the MediaStore and
 * other file-based ContentProviders.
 *
 * @param context The context.
 * @param uri The Uri to query.
 * @author paulburke
 * @source https://stackoverflow.com/a/20559175
 */
@TargetApi(Build.VERSION_CODES.KITKAT)
public static String getPath(final Context context, final Uri uri) {

    final boolean isKitKat = Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT;

    // DocumentProvider
    if (isKitKat && DocumentsContract.isDocumentUri(context, uri)) {
        // ExternalStorageProvider
        if (isExternalStorageDocument(uri)) {
            final String docId = DocumentsContract.getDocumentId(uri);
            final String[] split = docId.split(":");
            final String type = split[0];

            if ("primary".equalsIgnoreCase(type)) {
                return Environment.getExternalStorageDirectory() + "/" + split[1];
            }

            // TODO handle non-primary volumes
        }
        // DownloadsProvider
        else if (isDownloadsDocument(uri)) {

            final String id = DocumentsContract.getDocumentId(uri);
            final Uri contentUri = ContentUris.withAppendedId(
                    Uri.parse("content://downloads/public_downloads"), Long.valueOf(id));

            return getDataColumn(context, contentUri, null, null);
        }
        // MediaProvider
        else if (isMediaDocument(uri)) {
            final String docId = DocumentsContract.getDocumentId(uri);
            final String[] split = docId.split(":");
            final String type = split[0];

            Uri contentUri = null;
            if ("image".equals(type)) {
                contentUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;
            } else if ("video".equals(type)) {
                contentUri = MediaStore.Video.Media.EXTERNAL_CONTENT_URI;
            } else if ("audio".equals(type)) {
                contentUri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
            }

            final String selection = "_id=?";
            final String[] selectionArgs = new String[] {
                    split[1]
            };

            return getDataColumn(context, contentUri, selection, selectionArgs);
        }
    }
    // MediaStore (and general)
    else if ("content".equalsIgnoreCase(uri.getScheme())) {
        return getDataColumn(context, uri, null, null);
    }
    // File
    else if ("file".equalsIgnoreCase(uri.getScheme())) {
        return uri.getPath();
    }

    return null;
}

/**
 * Get the value of the data column for this Uri. This is useful for
 * MediaStore Uris, and other file-based ContentProviders.
 *
 * @param context The context.
 * @param uri The Uri to query.
 * @param selection (Optional) Filter used in the query.
 * @param selectionArgs (Optional) Selection arguments used in the query.
 * @return The value of the _data column, which is typically a file path.
 * @source https://stackoverflow.com/a/20559175
 */
public static String getDataColumn(Context context, Uri uri, String selection,
                                   String[] selectionArgs) {

    Cursor cursor = null;
    final String column = "_data";
    final String[] projection = {
            column
    };

    try {
        cursor = context.getContentResolver().query(uri, projection, selection, selectionArgs,
                null);
        if (cursor != null && cursor.moveToFirst()) {
            final int column_index = cursor.getColumnIndexOrThrow(column);
            return cursor.getString(column_index);
        }
    } finally {
        if (cursor != null)
            cursor.close();
    }
    return null;
}


/**
 * @param uri The Uri to check.
 * @return Whether the Uri authority is ExternalStorageProvider.
 * @source https://stackoverflow.com/a/20559175
 */
public static boolean isExternalStorageDocument(Uri uri) {
    return "com.android.externalstorage.documents".equals(uri.getAuthority());
}

/**
 * @param uri The Uri to check.
 * @return Whether the Uri authority is DownloadsProvider.
 * @source https://stackoverflow.com/a/20559175
 */
public static boolean isDownloadsDocument(Uri uri) {
    return "com.android.providers.downloads.documents".equals(uri.getAuthority());
}

/**
 * @param uri The Uri to check.
 * @return Whether the Uri authority is MediaProvider.
 * @source https://stackoverflow.com/a/20559175
 */
public static boolean isMediaDocument(Uri uri) {
    return "com.android.providers.media.documents".equals(uri.getAuthority());
}  

শেষ অবধি, এইচটিএমএল পৃষ্ঠাকে শোপিকারের সেই নতুন পদ্ধতিটি ট্রিগার করতে হবে (এ 4.4 এ থাকাকালীন)

<form id="form-upload" method="post" enctype="multipart/form-data">
    <input id="fileupload" name="fileupload" type="file" onclick="javascript:prepareForPicker();"/>
</form>
<script type="text/javascript">
function getAndroidVersion() {
    var ua = navigator.userAgent; 
    var match = ua.match(/Android\s([0-9\.]*)/);
    return match ? match[1] : false;
};
function prepareForPicker(){
    if(getAndroidVersion().indexOf("4.4") != -1){
        window.jsi.showPicker();
        return false;
    }
}

function CheckImage(name)
{
//Check to see if user wants to save I used some ajax to save the file if necesarry
}
</script>

দেরিতে উত্তরের জন্য ক্ষমা প্রার্থনা করুন, এর কোনও কাজের ডেমো রাখবেন না। এটি ছিল সেই সময়ে সরল গবেষণা research দুঃখিত :(
ডেকো

6

গুগলের নিজস্ব ব্রাউজার এই সমস্যার এমন একটি বিস্তৃত সমাধান সরবরাহ করে যে এটি তার নিজের শ্রেণীর জন্য ওয়ারেন্ট দেয়:

অ্যান্ড্রয়েড open.০.৪ এ ওপেনফাইলেচুজার প্রয়োগকরণ

অ্যান্ড্রয়েড Upload.০.৪ এ আপলোডহ্যান্ডলার ক্লাস


6

2019: এই কোডটি আমার পক্ষে কাজ করেছে (অ্যান্ড্রয়েড 5 - 9 এ পরীক্ষিত)।

package com.example.filechooser;

import android.app.Activity;
import android.content.Intent;
import android.net.Uri;
import android.net.http.SslError;
import android.os.Bundle;
import android.webkit.SslErrorHandler;
import android.webkit.ValueCallback;
import android.webkit.WebChromeClient;
import android.webkit.WebView;
import android.webkit.WebViewClient;


public class MainActivity extends Activity {

    // variables para manejar la subida de archivos
    private final static int FILECHOOSER_RESULTCODE = 1;
    private ValueCallback<Uri[]> mUploadMessage;

    // variable para manejar el navegador empotrado
    WebView mainWebView;


    @Override
    protected void onCreate(Bundle savedInstanceState) {

        super.onCreate(savedInstanceState);

        setContentView(R.layout.activity_main);

        // instanciamos el webview
        mainWebView = findViewById(R.id.main_web_view);

        // establecemos el cliente interno para que la navegacion no se salga de la aplicacion
        mainWebView.setWebViewClient(new MyWebViewClient());

        // establecemos el cliente chrome para seleccionar archivos
        mainWebView.setWebChromeClient(new MyWebChromeClient());

        // configuracion del webview
        mainWebView.getSettings().setJavaScriptEnabled(true);

        // cargamos la pagina
        mainWebView.loadUrl("https://example.com");
    }


    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent intent) {

        // manejo de seleccion de archivo
        if (requestCode == FILECHOOSER_RESULTCODE) {

            if (null == mUploadMessage || intent == null || resultCode != RESULT_OK) {
                return;
            }

            Uri[] result = null;
            String dataString = intent.getDataString();

            if (dataString != null) {
                result = new Uri[]{ Uri.parse(dataString) };
            }

            mUploadMessage.onReceiveValue(result);
            mUploadMessage = null;
        }
    }


    // ====================
    // Web clients classes
    // ====================

    /**
     * Clase para configurar el webview
     */
    private class MyWebViewClient extends WebViewClient {

        // permite la navegacion dentro del webview
        @Override
        public boolean shouldOverrideUrlLoading(WebView view, String url) {
            view.loadUrl(url);
            return true;
        }
    }


    /**
     * Clase para configurar el chrome client para que nos permita seleccionar archivos
     */
    private class MyWebChromeClient extends WebChromeClient {

        // maneja la accion de seleccionar archivos
        @Override
        public boolean onShowFileChooser(WebView webView, ValueCallback<Uri[]> filePathCallback, FileChooserParams fileChooserParams) {

            // asegurar que no existan callbacks
            if (mUploadMessage != null) {
                mUploadMessage.onReceiveValue(null);
            }

            mUploadMessage = filePathCallback;

            Intent i = new Intent(Intent.ACTION_GET_CONTENT);
            i.addCategory(Intent.CATEGORY_OPENABLE);
            i.setType("*/*"); // set MIME type to filter

            MainActivity.this.startActivityForResult(Intent.createChooser(i, "File Chooser"), MainActivity.FILECHOOSER_RESULTCODE );

            return true;
        }
    }

}

আশা আপনাকে সাহায্য করতে পারে।


অ্যাক্টিভিটিস রেজাল্ট পদ্ধতির জন্য এটি সুপার নিখোঁজ হওয়ার জন্য কল করেছে
সুমিতকোনজে

@ সুমিতকানোজে ইফ ব্লকের আগে কলটিকে সুপারের দিকে রাখার চেষ্টা করুন: সুপার.অনঅ্যাক্টিভিটিস রেজাল্ট (অনুরোধ কোড, রেজাল্ট কোড, উদ্দেশ্য);
সেবাসদেব

5

একটি সলিউশন পেয়েছি যা আমার পক্ষে কাজ করে! ফাইলটিতে আরও একটি বিধি যুক্ত করুন proguard-android.txt:

-keepclassmembers class * extends android.webkit.WebChromeClient {
     public void openFileChooser(...);
}

আপনি Android এর কোন সংস্করণ ব্যবহার করছেন?
পাঁচলনার্ন

4.4.4 সায়ানোজেডমড 11
ভিলিউসকে

4

অ্যান্ড্রয়েড 8 এর জন্য কোটলিন সমাধান:

private var mUploadMessage: ValueCallback<Uri>? = null
private var uploadMessage: ValueCallback<Array<Uri>>? = null

ধ্রুবক:

const val FILECHOOSER_RESULTCODE = 1
const val REQUEST_SELECT_FILE = 100

ওয়েবভিউ সেটআপ:

webView.webChromeClient = object : WebChromeClient() {
        override fun onPermissionRequest(request: PermissionRequest?) {
            Log.d("MainActivity", "onPermissionRequest")
            requestPermission(request)
        }


        // For Android 3.0+
        fun openFileChooser(uploadMsg: ValueCallback<*>, acceptType: String) {
            mUploadMessage = uploadMsg as ValueCallback<Uri>
            val i = Intent(Intent.ACTION_GET_CONTENT)
            i.addCategory(Intent.CATEGORY_OPENABLE)
            i.type = "*/*"
            this@MainActivity.startActivityForResult(
                    Intent.createChooser(i, "File Browser"),
                    FILECHOOSER_RESULTCODE)
        }

        //For Android 4.1
        fun openFileChooser(uploadMsg: ValueCallback<Uri>, acceptType: String, capture: String) {
            mUploadMessage = uploadMsg
            val i = Intent(Intent.ACTION_GET_CONTENT)
            i.addCategory(Intent.CATEGORY_OPENABLE)
            i.type = "image/*"
            this@MainActivity.startActivityForResult(Intent.createChooser(i, "File Chooser"), FILECHOOSER_RESULTCODE)

        }

        protected fun openFileChooser(uploadMsg: ValueCallback<Uri>) {
            mUploadMessage = uploadMsg
            val intent = Intent(Intent.ACTION_GET_CONTENT)
            intent.addCategory(Intent.CATEGORY_OPENABLE)
            intent.type = "*/*"
            startActivityForResult(Intent.createChooser(intent, "File Chooser"), FILECHOOSER_RESULTCODE)
        }

        override fun onShowFileChooser(webView: WebView?, filePathCallback: ValueCallback<Array<Uri>>?, fileChooserParams: FileChooserParams?): Boolean {
            uploadMessage?.onReceiveValue(null)
            uploadMessage = null

            uploadMessage = filePathCallback

            val intent = fileChooserParams!!.createIntent()
            try {
                startActivityForResult(intent, REQUEST_SELECT_FILE)
            } catch (e: ActivityNotFoundException) {
                uploadMessage = null
                Toast.makeText(applicationContext, "Cannot Open File Chooser", Toast.LENGTH_LONG).show()
                return false
            }

            return true
        }

    }

ওঅ্যাক্টিভিটি রেজাল্ট অংশ:

override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
        if (requestCode === REQUEST_SELECT_FILE) {
            if (uploadMessage == null)
                return
            print("result code = " + resultCode)
            var results: Array<Uri>? = WebChromeClient.FileChooserParams.parseResult(resultCode, data)
            uploadMessage?.onReceiveValue(results)
            uploadMessage = null
        }
    } else if (requestCode === FILECHOOSER_RESULTCODE) {
        if (null == mUploadMessage)
            return
        // Use MainActivity.RESULT_OK if you're implementing WebView inside Fragment
        // Use RESULT_OK only if you're implementing WebView inside an Activity
        val result = if (intent == null || resultCode !== RESULT_OK) null else intent.data
        mUploadMessage?.onReceiveValue(result)
        mUploadMessage = null
    } else
        Toast.makeText(applicationContext, "Failed to Upload Image", Toast.LENGTH_LONG).show()
}

দয়া করে মনোযোগ দিন যে আমাদের উদ্দেশ্য ভেরিয়েবল "ডেটা" বলে।


1
ধন্যবাদ! আমি আপনাকে একটি বিয়ার
রাফায়েল রউইজ মুউজ

2

আপনার প্রস্তাবনা জন্য ধন্যবাদ। আসল সমস্যাটি হ'ল - ওয়েবভিউ আমাকে ফাইল আপলোড ডায়লগ বাক্সটি উপস্থাপন করছে না যেখানে আমি আপলোড করার জন্য একটি ফাইল বেছে নিতে পারি । সুতরাং, প্রথমে আমার একটি সমাধান দরকার যা আমাকে ফাইল আপলোড ডায়লগ বাক্সটি প্রদর্শন করতে সক্ষম করে, তারপরে আমি আপনার প্রস্তাবিত লিঙ্কগুলি উল্লেখ করতে পারি।
ব্যবহারকারী 741148


থেক্সএক্স 4 আপনার পরামর্শ প্রজ্ঞা। কিছুই কাজ করে না ... আসলে আমি যা পর্যবেক্ষণ করেছি তা হ'ল ওয়েবক্রোমক্লিয়েন্টের ওপেনফাইলেচুসার (ভ্যালু ক্যালব্যাক <ইউরি>) পদ্ধতি অ্যান্ড্রয়েড ২.২ এ কাজ করে, যেখানে ইউআইতে একটি <ইনপুট টাইপ = "ফাইল"> ক্ষেত্রে ক্লিক করা থেকে ফাইল নির্বাচন সহজতর করে গ্যালারী, তবে, অ্যান্ড্রয়েড 3.0.০ ... সেখানেই কাজ করে না, আমি ফ্ল্যাশ ফাইল আপলোড দিয়ে ওয়েবভিউতে ফাইল আপলোড করার চেষ্টা করার কথা ভেবেছিলাম, তবে অ্যান্ড্রয়েড 3.0.০ এমুলেটর ফ্ল্যাশ ডিফল্টরূপে ইনস্টল করা হয়নি এবং এটি ইনস্টল করতে বাজার অ্যাপ্লিকেশন উপলব্ধ নেই এবং জটিলতায় ... উপসংহারে, সমাধানগুলি দেওয়া দিয়ে অসম্ভব বলে মনে হচ্ছে !
ব্যবহারকারী 741148

1

কিটকেটে আপনি স্টোরেজ অ্যাক্সেস ফ্রেমওয়ার্কটি ব্যবহার করতে পারেন।

স্টোরেজ অ্যাক্সেস ফ্রেমওয়ার্ক / ক্লায়েন্ট অ্যাপ লিখন


"হতে পারে"? আপনি কি আপনার পোস্টের জন্য আরও প্রসঙ্গ সরবরাহ করতে পারেন? দয়া করে, কীভাবে উত্তর দেবেন তা পড়ুন
brasofilo

ওয়েব ভিউ ব্যবহার করে এন্ড্রয়েড ৪.৪.২ এ এটি প্রয়োগ করার জন্য কি নমুনা কোড রয়েছে?
বাবু কে

1

ওয়েবভিউ - একক এবং একাধিক ফাইল চয়ন করুন

এই কোডটি প্রয়োগ করতে আপনার দুই মিনিটের প্রয়োজন:

build.gradle

implementation 'com.github.angads25:filepicker:1.1.1'

জাভা কোড:

import android.annotation.SuppressLint;
import android.app.Activity;
import android.content.DialogInterface;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.graphics.Bitmap;
import android.net.Uri;
import android.os.Build;
import android.os.Bundle;
import android.support.annotation.NonNull;
import android.util.Log;
import android.view.KeyEvent;
import android.view.View;
import android.webkit.ValueCallback;
import android.webkit.WebChromeClient;
import android.webkit.WebSettings;
import android.webkit.WebView;
import android.webkit.WebViewClient;
import android.widget.ProgressBar;
import android.widget.Toast;

import com.bivoiclient.utils.Constants;
import com.github.angads25.filepicker.controller.DialogSelectionListener;
import com.github.angads25.filepicker.model.DialogConfigs;
import com.github.angads25.filepicker.model.DialogProperties;
import com.github.angads25.filepicker.view.FilePickerDialog;

import java.io.File;

public class WebBrowserScreen extends Activity {

    private WebView webView;
    private ValueCallback<Uri[]> mUploadMessage;
    private FilePickerDialog dialog;
    private String LOG_TAG = "DREG";
    private Uri[] results;

    @SuppressLint("SetJavaScriptEnabled")
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_complain);

        webView = findViewById(R.id.webview);
        WebSettings webSettings = webView.getSettings();
        webSettings.setAppCacheEnabled(true);
        webSettings.setCacheMode(WebSettings.LOAD_CACHE_ELSE_NETWORK);
        webSettings.setJavaScriptEnabled(true);
        webSettings.setLoadWithOverviewMode(true);
        webSettings.setAllowFileAccess(true);
        webView.setWebViewClient(new PQClient());
        webView.setWebChromeClient(new PQChromeClient());
        if (Build.VERSION.SDK_INT >= 19) {
            webView.setLayerType(View.LAYER_TYPE_HARDWARE, null);
        } else {
            webView.setLayerType(View.LAYER_TYPE_SOFTWARE, null);
        }

        webView.loadUrl(Constants.COMPLAIN_URL);

    }

    private void openFileSelectionDialog() {

        if (null != dialog && dialog.isShowing()) {
            dialog.dismiss();
        }

        //Create a DialogProperties object.
        final DialogProperties properties = new DialogProperties();

        //Instantiate FilePickerDialog with Context and DialogProperties.
        dialog = new FilePickerDialog(WebBrowserScreen.this, properties);
        dialog.setTitle("Select a File");
        dialog.setPositiveBtnName("Select");
        dialog.setNegativeBtnName("Cancel");
        properties.selection_mode = DialogConfigs.MULTI_MODE; // for multiple files
//        properties.selection_mode = DialogConfigs.SINGLE_MODE; // for single file
        properties.selection_type = DialogConfigs.FILE_SELECT;

        //Method handle selected files.
        dialog.setDialogSelectionListener(new DialogSelectionListener() {
            @Override
            public void onSelectedFilePaths(String[] files) {
                results = new Uri[files.length];
                for (int i = 0; i < files.length; i++) {
                    String filePath = new File(files[i]).getAbsolutePath();
                    if (!filePath.startsWith("file://")) {
                        filePath = "file://" + filePath;
                    }
                    results[i] = Uri.parse(filePath);
                    Log.d(LOG_TAG, "file path: " + filePath);
                    Log.d(LOG_TAG, "file uri: " + String.valueOf(results[i]));
                }
                mUploadMessage.onReceiveValue(results);
                mUploadMessage = null;
            }
        });
        dialog.setOnCancelListener(new DialogInterface.OnCancelListener() {
            @Override
            public void onCancel(DialogInterface dialogInterface) {
                if (null != mUploadMessage) {
                    if (null != results && results.length >= 1) {
                        mUploadMessage.onReceiveValue(results);
                    } else {
                        mUploadMessage.onReceiveValue(null);
                    }
                }
                mUploadMessage = null;
            }
        });
        dialog.setOnDismissListener(new DialogInterface.OnDismissListener() {
            @Override
            public void onDismiss(DialogInterface dialogInterface) {
                if (null != mUploadMessage) {
                    if (null != results && results.length >= 1) {
                        mUploadMessage.onReceiveValue(results);
                    } else {
                        mUploadMessage.onReceiveValue(null);
                    }
                }
                mUploadMessage = null;
            }
        });

        dialog.show();

    }

    public class PQChromeClient extends WebChromeClient {

        @Override
        public boolean onShowFileChooser(WebView webView, ValueCallback<Uri[]> filePathCallback, FileChooserParams fileChooserParams) {
            // Double check that we don't have any existing callbacks
            if (mUploadMessage != null) {
                mUploadMessage.onReceiveValue(null);
            }
            mUploadMessage = filePathCallback;

            openFileSelectionDialog();

            return true;
        }

    }

    //Add this method to show Dialog when the required permission has been granted to the app.
    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String permissions[], @NonNull int[] grantResults) {
        switch (requestCode) {
            case FilePickerDialog.EXTERNAL_READ_PERMISSION_GRANT: {
                if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                    if (dialog != null) {
                        openFileSelectionDialog();
                    }
                } else {
                    //Permission has not been granted. Notify the user.
                    Toast.makeText(WebBrowserScreen.this, "Permission is Required for getting list of files", Toast.LENGTH_SHORT).show();
                }
            }
        }
    }

    public boolean onKeyDown(int keyCode, KeyEvent event) {
        // Check if the key event was the Back button and if there's history
        if ((keyCode == KeyEvent.KEYCODE_BACK) && webView.canGoBack()) {
            webView.goBack();
            return true;
        }
        // If it wasn't the Back key or there's no web page history, bubble up to the default
        // system behavior (probably exit the activity)

        return super.onKeyDown(keyCode, event);
    }


    public class PQClient extends WebViewClient {
        ProgressBar progressDialog;

        public boolean shouldOverrideUrlLoading(WebView view, String url) {

            // If url contains mailto link then open Mail Intent
            if (url.contains("mailto:")) {

                // Could be cleverer and use a regex
                //Open links in new browser
                view.getContext().startActivity(
                        new Intent(Intent.ACTION_VIEW, Uri.parse(url)));

                // Here we can open new activity

                return true;

            } else {
                // Stay within this webview and load url
                view.loadUrl(url);
                return true;
            }
        }

        // Show loader on url load
        public void onPageStarted(WebView view, String url, Bitmap favicon) {
            // Then show progress  Dialog
            // in standard case YourActivity.this
            if (progressDialog == null) {
                progressDialog = findViewById(R.id.progressBar);
                progressDialog.setVisibility(View.VISIBLE);
            }
        }

        // Called when all page resources loaded
        public void onPageFinished(WebView view, String url) {
            webView.loadUrl("javascript:(function(){ " +
                    "document.getElementById('android-app').style.display='none';})()");

            try {
                // Close progressDialog
                progressDialog.setVisibility(View.GONE);
            } catch (Exception exception) {
                exception.printStackTrace();
            }
        }
    }

}

এই লাইনটি ত্রুটি আমদানি দেখায় com.bivoiclient.utils.Constants;
মোঃ আশিকুর রহমান

এই আমদানিটি সরান এবং এতে আপনার ইউআরএল সেট করুন: ওয়েবভিউ.ওলডআরল (" google.com" );
আহমাদুল্লাহ সৈকত

হাই আহমাদউল্লাহ সৈকত, এই ধরনের প্রচেষ্টা করার জন্য, এটি কাস্টম হিসাবে কাজ করে ... ডিফল্টরূপে কাজ করে না। কোনও ব্যবহারকারী যদি ওয়েব ভিউতে ডিফল্ট সংযুক্তি বিকল্পগুলি খুলতে চান তবে কীভাবে সে তা করতে পারে। এটি প্রায় কাজ।
তারিত রায়

0

আমি আন্দ্রিওডে নতুন এবং এ নিয়েও লড়াই করেছি। গুগল রেফারেন্স গাইড ওয়েবভিউ অনুসারে।

ডিফল্টরূপে, কোনও ওয়েবভিউ ব্রাউজারের মতো উইজেট সরবরাহ করে না, জাভাস্ক্রিপ্ট সক্ষম করে না এবং ওয়েব পৃষ্ঠার ত্রুটি উপেক্ষা করা হয়। যদি আপনার লক্ষ্যটি কেবলমাত্র আপনার ইউআই এর অংশ হিসাবে কিছু HTML প্রদর্শন করা হয় তবে এটি সম্ভবত ঠিক আছে; এটি পড়ার বাইরে ব্যবহারকারীর ওয়েবপৃষ্ঠের সাথে ইন্টারঅ্যাক্ট করার প্রয়োজন হবে না, এবং ওয়েব পৃষ্ঠাকে ব্যবহারকারীর সাথে ইন্টারঅ্যাক্ট করার প্রয়োজন হবে না। আপনি যদি সত্যিই একটি পূর্ণ-বিকাশযুক্ত ওয়েব ব্রাউজার চান, তবে আপনি সম্ভবত ব্রাউজার অ্যাপ্লিকেশনটি কোনও ওয়েবউইউ দিয়ে না দেখানোর পরিবর্তে একটি URL ইন্টেন্টের সাহায্যে আবেদন করতে চান।

মেইনএ্যাকটিভিটি.জভাতে আমি এক্সিকিউট করা উদাহরণ কোড।

 Uri uri = Uri.parse("https://www.example.com");
 Intent intent = new Intent(Intent.ACTION_VIEW, uri);
 startActivity(intent);

Excuted

package example.com.myapp;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.webkit.WebView;
import android.webkit.WebViewClient;
import android.content.Intent;
import android.net.Uri;

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        Uri uri = Uri.parse("http://www.example.com/");
        Intent intent = new Intent(Intent.ACTION_VIEW, uri);
        startActivity(intent);
        getSupportActionBar().hide();
    }}

0

আমি প্রায় এক মাস ধরে এই বিষয়ে গবেষণা করেছি। সবকিছু ব্যর্থ হয়েছে। একাধিক ওয়েবসাইটে দেখা সমস্ত কোডগুলি ভাল কাজ করে না। তবে এখানে সবচেয়ে ভাল সমাধান বিদ্যমান

https://github.com/chiclaim/android-webview-upload-file

ধাপ

1) ক্লোন বা ডাউনলোড এ ক্লিক করুন

2) আপনার স্থানীয় ডিরেক্টরিতে জিপ ফ্লাইট পান

3) জিপ ফাইলটি আনজিপ করুন

4) অ্যান্ড্রয়েড স্টুডিও খুলুন

5) গোটো ফাইল ----> খুলুন ---> ডিরেক্টরিতে নেভিগেট করুন যেখানে আপনি বিষয়বস্তুগুলি আনজিপ করেছেন।

6) ওয়েবভিউ.লোডউইআরএলটিতে প্রয়োজনীয় ওয়েব ইউআরএল পরিবর্তন করুন ("আপনার ইউআরএল এখানে"); মেইনএকটিভিটি.জভাতে

7) সংস্করণ অ্যান্ড্রয়েড স্টুডিও 3.4.2 সঙ্গে ভাল কাজ করে

আমাদের সাইট ব্যবহার করে, আপনি স্বীকার করেছেন যে আপনি আমাদের কুকি নীতি এবং গোপনীয়তা নীতিটি পড়েছেন এবং বুঝতে পেরেছেন ।
Licensed under cc by-sa 3.0 with attribution required.