从通知中获取 url 并加载到 Fragment 的 webview 的最佳方法?

Posted

技术标签:

【中文标题】从通知中获取 url 并加载到 Fragment 的 webview 的最佳方法?【英文标题】:Best method to get url from notification and load into Fragment's webview? 【发布时间】:2013-05-16 20:35:51 【问题描述】:

所以基本上我刚接触android,我正在制作一个应用程序,它可以发送从服务器发送的实时通知。

现在我希望用户单击通知时会在我的 web 视图中打开一个网页。 它是一个基于网络的应用程序。

我使用 3 个选项卡在 3 个碎片之间切换。 每个片段都有不同的 webview。

现在这是我的 GCMintentService.java:

   package blah.blah;


import static blah.blah.CommonUtilities.SENDER_ID;

import static blah.blah.CommonUtilities.displayMessage;
import blah.blah.R;
import blah.blah.R.drawable;
import blah.blah.R.string;

import android.app.Notification;
import android.app.NotificationManager;
import android.app.PendingIntent;
import android.content.Context;
import android.content.Intent;
import android.util.Log;

import com.google.android.gcm.GCMBaseIntentService;

public class GCMIntentService extends GCMBaseIntentService 

    private static final String TAG = "GCMIntentService";

    public GCMIntentService() 
        super(SENDER_ID);
    

    /**
     * Method called on device registered
     **/
    @Override
    protected void onRegistered(Context context, String registrationId) 
        Log.i(TAG, "Toestel geregistreerd: regId = " + registrationId);
        displayMessage(context, "Je toestel is geregistreerd");
        Log.d("NAME", NotificationMain.name);
        ServerUtilities.register(context, NotificationMain.name, NotificationMain.klas, registrationId);
    

    /**
     * Method called on device un registred
     * */
    @Override
    protected void onUnregistered(Context context, String registrationId) 
        Log.i(TAG, "Toestel nog niet geregistreerd!");
        displayMessage(context, getString(R.string.gcm_unregistered));
        ServerUtilities.unregister(context, registrationId);
    

    /**
     * Method called on Receiving a new message
     * */
    @Override
    protected void onMessage(Context context, Intent intent) 
        Log.i(TAG, "Ontvangen bericht");
        String message = intent.getExtras().getString("price");

        displayMessage(context, message);
        // notifies user
        generateNotification(context, message);
    

    /**
     * Method called on receiving a deleted message
     * */
    @Override
    protected void onDeletedMessages(Context context, int total) 
        Log.i(TAG, "Received deleted messages notification");
        String message = getString(R.string.gcm_deleted, total);
        displayMessage(context, message);
        // notifies user
        generateNotification(context, message);
    

    /**
     * Method called on Error
     * */
    @Override
    public void onError(Context context, String errorId) 
        Log.i(TAG, "Received error: " + errorId);
        displayMessage(context, getString(R.string.gcm_error, errorId));
    

    @Override
    protected boolean onRecoverableError(Context context, String errorId) 
        // log message
        Log.i(TAG, "Received recoverable error: " + errorId);
        displayMessage(context, getString(R.string.gcm_recoverable_error,
                errorId));
        return super.onRecoverableError(context, errorId);
    

    /**
     * Issues a notification to inform the user that server has sent a message.
     */
    private static void generateNotification(Context context, String message) 
        int icon = R.drawable.ic_launcher;
        long when = System.currentTimeMillis();
        NotificationManager notificationManager = (NotificationManager)
                context.getSystemService(Context.NOTIFICATION_SERVICE);
        Notification notification = new Notification(icon, message, when);

        String title = context.getString(R.string.app_name);

        Intent notificationIntent = new Intent(context, MyFragment.class);
        // set intent so it does not start a new activity
        notificationIntent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP |
                Intent.FLAG_ACTIVITY_SINGLE_TOP);
        PendingIntent intent =
                PendingIntent.getActivity(context, 0, notificationIntent, 0);
        notification.setLatestEventInfo(context, title, message, intent);
        notification.flags |= Notification.FLAG_AUTO_CANCEL;

        // Play default notification sound
        notification.defaults |= Notification.DEFAULT_SOUND;

        //notification.sound = Uri.parse("android.resource://" + context.getPackageName() + "your_sound_file_name.mp3");

        // Vibrate if vibrate is enabled
        notification.defaults |= Notification.DEFAULT_VIBRATE;
        notificationManager.notify(0, notification);      

    


一个主要的活动包含 3 个具有所有不同 web 视图的片段。 我从一个名为 FragmentAdapter.java 的数组中获取 url,它扩展了 FragmentAdapter。

完美运行...在这里 是fragmentAdapter:

    package blah.blah;

import blah.blah.MyFragment;


import android.os.Bundle;
import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentManager;
import android.support.v4.app.FragmentPagerAdapter;

public class MyFragmentPagerAdapter extends FragmentPagerAdapter


    String[] toVisit=
        "www.google.com",
        "www.stackoverlow.com",
        "www.lorumipsum.com",
        ;

    final int PAGE_COUNT = 3;

    public MyFragmentPagerAdapter(FragmentManager fm) 
        super(fm);
    
    @Override
      public Fragment getItem(int position) 
        // Here is where all the magic of the adapter happens
        // As you can see, this is really simple.
        return MyFragment.newInstance(toVisit[position]);
    
    @Override
    public int getCount()      
        return PAGE_COUNT;
    
    @Override
    public CharSequence getPageTitle(int position)         
        if(position == 0)
        
            return "Klassen";
        
        else if(position == 1)
        
            return "Docenten";
        
        else
        
            return "Lokalen";
           
       

但是如何根据从服务器检索到的信息加载不同的 url? , GCM 推送是通过 php 发送的,带有通常的代码。

应该是有意图的东西。PushExtra("google.com, http://google.com") // 例如 一切都不起作用。我希望我的问题很清楚。

感谢所有帮助! 只需分享您的所有提示和想法:P。

如果有用的话,还有我的 MyFragment.java:

    package blah.blah;


import blah.blah.R;
import blah.blah.R.id;
import blah.blah.R.layout;
import android.content.Intent;
import android.os.Bundle;
import android.support.v4.app.Fragment;
import android.view.LayoutInflater;
import android.view.MenuItem;
import android.view.View;
import android.view.ViewGroup;
import android.webkit.WebView;
import android.webkit.WebViewClient;
import android.widget.ProgressBar;


public class MyFragment extends Fragment

    WebView browser; 
    String url;
    private Bundle webViewBundle;
    Intent extras = getActivity().getIntent();

    @Override
    public View onCreateView(LayoutInflater inflater, 
         ViewGroup container, Bundle savedInstanceState) 

        View view=inflater.inflate(
            R.layout.myfragment_layout, 
            container, 
            false);

        final ProgressBar spinner = (ProgressBar)view.findViewById(R.id.progress);

        browser=(WebView)view.findViewById(R.id.webView1);
        browser.getSettings().setjavascriptEnabled(true);
        browser.setWebViewClient(new WebViewClient() 

            public void onReceivedError(WebView view, int errorCode, String description, String failingUrl) 
                browser.loadUrl("file:///android_asset/geeninternet.html");

            
            @SuppressWarnings("unused")
            public void onProgressChanged(final WebView view, final int progress)   
            
                if(progress == 100)
                    spinner.setVisibility(View.GONE);
              
        );

        browser.loadUrl(url); 

        // Just load whatever URL this fragment is
        // created with.
        return view;
    
    // This is the method the pager adapter will use
    // to create a new fragment
    public static Fragment newInstance(String url)
    
        MyFragment f=new MyFragment();
        f.url=url;
        return f;
    
    // Met browser;
    public boolean onOptionsItemSelected(MenuItem item)
    
        switch (item.getItemId())
        
        case R.id.menu_refresh:
        browser.loadUrl(url);
            return true;
        default:
            return super.onOptionsItemSelected(item);
        
    

    /**
     * Sla webview op
     */
    @Override
    public void onPause()
    
        super.onPause();

        webViewBundle = new Bundle();
        browser.saveState(webViewBundle);
    

    /**
     * Herstel staat van webview
     */
    @Override
    public void onActivityCreated(Bundle savedInstanceState)
    
        super.onActivityCreated(savedInstanceState);

        if (webViewBundle != null)
        
            browser.restoreState(webViewBundle);
        
    

【问题讨论】:

【参考方案1】:
@Override
protected void onMessage(Context context, Intent intent) 
    Log.i(TAG, "Ontvangen bericht");
    String message = intent.getExtras().getString("price");

    displayMessage(context, message);
    // notifies user
    generateNotification(context, message);

在此代码中,您只从通知负载中获取一个值 - price 键的值。

您没有包含服务器代码,所以我不知道您是否发送了带有通知的 URL。如果这样做,您应该从onMessage 方法中的intent 获取其值,并将其传递给打开网络视图的代码。如果不这样做,则应更改服务器代码以将 URL 包含在通知负载中。

编辑:

为了将 URL 传递给您的片段:

    onMessage() 中的额外意图中获取 URL。

    将其传递给generateNotification

    将 URL 作为额外内容添加到在点击通知时将启动 Activity 的 Intent 中:notificationIntent.putExtra ("url",url)

    在片段中,在初始化 web 视图的代码中,从启动包含片段 (getActivity().getIntent().getExtras().getString("url") 的活动的意图的附加内容中获取 URL。

    李>

EDIT2:

    您应该在onActivityCreated() 中加载网络视图,而不是在onCreateView() 中。 onCreateView() 在活动创建之前被调用,所以getActivity() 将返回 null。

    不要按原样使用该代码:String test = (getActivity().getIntent().getExtras().getString("url")); 确保 getActivity() 不返回 null。我相信当getActivity() 不为null 时,您可以假设getIntent()getExtras() 不会返回null,但检查起来更安全。如果活动不是从通知中打开的,getString("url") 将返回 null。因此,在加载 web-view 之前,您应该确保它有一个值。

    通知标志可能有问题:

// set intent so it does not start a new activity
notificationIntent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP |
        Intent.FLAG_ACTIVITY_SINGLE_TOP);

如果这样的活动已经存在(例如,如果应用程序已经在运行),通知似乎不会打开新活动。在这种情况下,您的 onCreateView()onActivityCreated() 的片段代码将不会被调用,并且 URL 也不会被加载。您应该更改标志以便始终创建新活动,或者您应该将加载 Web 视图的逻辑作为通知的结果放在其他地方 - 可能在 onResume()

【讨论】:

谢谢你的回答!我在几分钟内尝试一下:)我对 java 很陌生,所以我使用了这个基础:androidhive.info/2012/10/… 我使用这种方法将数据发送到所需的设备,我刚刚编辑过,但它是同一个系统。 我可以成功从服务器获取url;我可以用 java 收到: 检查我编辑的答案(我会在几分钟内删除它,否则我无法以正确的方式添加代码..),............我现在将编辑后的代码添加到我的问题中。谢谢提示。 我真的很抱歉,但我无法得到它......它只是没有注意到 putExtra......,我完全不熟悉这个,所以如果我真的很感激可以获得 10 分钟的时间,并且您可以将 myFragment.java 编辑为对我有用的示例,如果我自己加载应用程序,我会得到一个空指针异常,如果我从通知中加载它,它只需要 mainactivity.java ... .. MainActivity 是包含你的 Fragment 的 Activity 吗?

以上是关于从通知中获取 url 并加载到 Fragment 的 webview 的最佳方法?的主要内容,如果未能解决你的问题,请参考以下文章

如何从 iframe 获取当前页面加载的 url [重复]

从 Adapter 向 Fragment 发送数据并在 Fragment 中获取

将图像保存到数据库并从服务器加载

Javasrcipt中从一个url或者从一个字符串中获取参数值得方法

从输入URL到页面加载完成的过程中都发生了什么事情?

从 url 获取哈希并在页面加载时单击带有数据属性的链接