Android에서 클래스 파일에서 프로그래밍 방식으로 탐색 서랍 헤더 이미지와 이름을 설정하는 방법은 무엇입니까?


79

안드로이드 스튜디오 1.4.1에서 기본적으로 새로운 Navigation Drawer 프로젝트를 생성했는데,이 프로젝트에는 네비게이션 헤더 이미지와 이름을위한 nav_header_main.xml 파일이 있습니다. 이 이미지와 이름은 기본 수업 활동에서 프로그래밍 방식으로 설정되어야합니다. 이렇게하는 방법을 많이 시도했지만 앱이 충돌합니다.

nav_header_main.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout        
android:layout_width="match_parent"
android:id="@+id/headerView"
android:layout_height="@dimen/nav_header_height"
android:background="@drawable/side_nav_bar"
android:gravity="bottom"
android:orientation="vertical"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
android:theme="@style/ThemeOverlay.AppCompat.Dark">

<ImageView
    android:id="@+id/imageView"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:paddingTop="@dimen/nav_header_vertical_spacing"
    android:src="@android:drawable/sym_def_app_icon" />

<TextView
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:paddingTop="@dimen/nav_header_vertical_spacing"
    android:text="Android Studio"
    android:textAppearance="@style/TextAppearance.AppCompat.Body1" />

<TextView
    android:id="@+id/textView"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="android.studio@android.com" />

</LinearLayout>

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<android.support.v4.widget.DrawerLayout            

xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/drawer_layout"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:fitsSystemWindows="true"
tools:openDrawer="start">

<include
    layout="@layout/app_bar_main"
    android:layout_width="match_parent"
    android:layout_height="match_parent" />

<android.support.design.widget.NavigationView
    android:id="@+id/nav_view"
    android:layout_width="wrap_content"
    android:layout_height="match_parent"
    android:layout_gravity="start"
    android:fitsSystemWindows="true"
    app:headerLayout="@layout/nav_header_main"
    app:menu="@menu/activity_main_drawer" />

    </android.support.v4.widget.DrawerLayout>

MainActivity.Class

import android.os.Bundle;
import android.support.design.widget.FloatingActionButton;
import android.support.design.widget.NavigationView;
import android.support.design.widget.Snackbar;
import android.support.v4.view.GravityCompat;
import android.support.v4.widget.DrawerLayout;
import android.support.v7.app.ActionBarDrawerToggle;
import android.support.v7.app.AppCompatActivity;
import android.support.v7.widget.Toolbar;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.LinearLayout;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity
        implements NavigationView.OnNavigationItemSelectedListener {

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

        LinearLayout headerImageView= (LinearLayout) findViewById(R.id.headerView);


        Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
        setSupportActionBar(toolbar);

        FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
        fab.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                Snackbar.make(view, "Replace with your own action", Snackbar.LENGTH_LONG)
                        .setAction("Action", null).show();
            }
        });

        DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
        ActionBarDrawerToggle toggle = new ActionBarDrawerToggle(
                this, drawer, toolbar, R.string.navigation_drawer_open, R.string.navigation_drawer_close);
        drawer.setDrawerListener(toggle);
        toggle.syncState();

        NavigationView navigationView = (NavigationView) findViewById(R.id.nav_view);
        navigationView.setNavigationItemSelectedListener(this);

    }

    @Override
    public void onBackPressed() {
        DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
        if (drawer.isDrawerOpen(GravityCompat.START)) {
            drawer.closeDrawer(GravityCompat.START);
        } else {
            super.onBackPressed();
        }
    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.main, menu);
        return true;
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        // Handle action bar item clicks here. The action bar will
        // automatically handle clicks on the Home/Up button, so long
        // as you specify a parent activity in AndroidManifest.xml.
        int id = item.getItemId();

        //noinspection SimplifiableIfStatement
        if (id == R.id.action_settings) {
            Toast.makeText(getApplicationContext(),"working",Toast.LENGTH_LONG).show();
            return true;
        }

        return super.onOptionsItemSelected(item);
    }

    @SuppressWarnings("StatementWithEmptyBody")
    @Override
    public boolean onNavigationItemSelected(MenuItem item) {
        // Handle navigation view item clicks here.
        int id = item.getItemId();

        if (id == R.id.nav_camara) {
            // Handle the camera action
        } else if (id == R.id.nav_gallery) {

        } else if (id == R.id.nav_slideshow) {

        } else if (id == R.id.nav_manage) {

        } else if (id == R.id.nav_share) {

        } else if (id == R.id.nav_send) {

        }

        DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
        drawer.closeDrawer(GravityCompat.START);

        return true;
    }
}

답변:


243
NavigationView navigationView = (NavigationView) findViewById(R.id.nav_view);
View hView =  navigationView.getHeaderView(0);
TextView nav_user = (TextView)hView.findViewById(R.id.nav_name);
nav_user.setText(user);

이 도움을 바랍니다!


2
navigationView.getHeaderView (0); 왜 0을 사용합니까?
Lahiru Liyanage

51

버그 190226 에서 언급했듯이 버전 23.1.0부터 헤더 레이아웃보기를 가져 오는 경우 : navigationView.findViewById(R.id.navigation_header_text)더 이상 작동하지 않습니다.

해결 방법은 헤더 뷰를 프로그래밍 방식으로 확장하고 확장 된 헤더 뷰에서 ID로 뷰를 찾는 것입니다.

예를 들면 :

View headerView = navigationView.inflateHeaderView(R.layout.navigation_header);
headerView.findViewById(R.id.navigation_header_text);

이상적으로는 방법이 있어야 getHeaderView()하지만 이미 제안되어 있으므로 디자인 지원 라이브러리의 기능 릴리스에서 릴리스 될 때까지보고 기다립니다.


당신은 또한 텍스트보기, 이미지보기를 추가하는 방법을 말 해주세요 수
칼라이

1
Dude, 답변에서 언급했듯이 inflateHeaderView()메서드를 사용하여 헤더 레이아웃 (이미지보기, 텍스트보기 또는 헤더 레이아웃에서 원하는 모든보기 포함)을 확장 할 수 있으며 다음을 사용하여 동일한 확장 레이아웃에서보기를 찾을 수 있습니다.findViewById()
Paresh Mayani

NavigationView navigationView = null; 보기 headerView = navigationView.inflateHeaderView (R.layout.nav_header_main); TextView tv = (TextView) headerView.findViewById (R.id.textView); tv.setText ( "확인"); 이 내 코드에 있는데도 응용 프로그램 충돌
칼라이

11
주어진 예제로 작업하고 android.support.design.widget.NavigationView app:headerLayout="@layout/nav_header_main"이미 프로그래밍 방식으로 팽창했기 때문에 줄을 제거하십시오
MilapTank

@milapTank 주석은 XML에서 그에 의해 제안 된 줄을 제거 할 때까지 헤더보기가 두 번 부풀려지면서 도움이되었습니다.
Taimur Khan 2016 년

25

레이아웃을 확장하여 코드를 사용하여 xml 추가에 헤더를 추가하지 마십시오.

View hView =  navigationView.inflateHeaderView(R.layout.nav_header_main);
ImageView imgvw = (ImageView)hView.findViewById(R.id.imageView);
TextView tv = (TextView)hView.findViewById(R.id.textview);
imgvw .setImageResource();
tv.settext("new text");

예, 이것은 쉽게 결과를 얻고 있습니다, 감사합니다 bhai.
Sajid khan

7

Kotlin에서

    val hView = nav_view.getHeaderView(0)
    val textViewName = hView.findViewById(R.id.textViewName) as TextView
    val textViewEmail = hView.findViewById(R.id.textViewEmail) as TextView
    val imgvw = hView.findViewById(R.id.imageView) as ImageView
    imgvw.setImageResource(R.drawable.ic_menu_gallery)

6

먼저 다음과 같이 MainActivity (또는 호출 활동)의 탐색 창에 액세스해야합니다.

    NavigationView navigationView = (NavigationView) findViewById(R.id.nav_view);

그런 다음 레이아웃이 MainActivity에서 프로그래밍 방식으로 확장되므로 activity_main.xml에서 헤더 레이아웃을 제거해야합니다. activity_main.xml은 다음과 같아야합니다.

<?xml version="1.0" encoding="utf-8"?>
<android.support.v4.widget.DrawerLayout            

xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/drawer_layout"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:fitsSystemWindows="true"
tools:openDrawer="start">

<include
    layout="@layout/app_bar_main"
    android:layout_width="match_parent"
    android:layout_height="match_parent" />

<android.support.design.widget.NavigationView
    android:id="@+id/nav_view"
    android:layout_width="wrap_content"
    android:layout_height="match_parent"
    android:layout_gravity="start"
    android:fitsSystemWindows="true"
    app:menu="@menu/activity_main_drawer" />

    </android.support.v4.widget.DrawerLayout>

그런 다음 MainActivity에서 nav_header_main 레이아웃을 확장하고 해당 뷰 (이 경우 ImageView 및 TextView)에 액세스합니다.

//inflate header layout
View navView =  navigationView.inflateHeaderView(R.layout.nav_header_main);
//reference to views
ImageView imgvw = (ImageView)navView.findViewById(R.id.imageView);
TextView tv = (TextView)navView.findViewById(R.id.textview);
//set views
imgvw.setImageResource(R.drawable.your_image);
tv.setText("new text");

navigationView.setNavigationItemSelectedListener(this);

여기에서 더 많은 것을 읽을 수 있습니다


5

다음은 완벽하게 작동하는 코드입니다. activity_main.xml의 NavigationView 태그에 헤더를 추가하지 마십시오.

<include
    layout="@layout/app_bar_main"
    android:layout_width="match_parent"
    android:layout_height="match_parent" />

<android.support.design.widget.NavigationView
    android:id="@+id/nav_view"
    android:layout_width="wrap_content"
    android:layout_height="match_parent"
    android:layout_gravity="start"
    android:fitsSystemWindows="true"
    app:menu="@menu/activity_main_drawer"
    app:itemBackground="@drawable/active_drawer_color" />

아래 코드를 사용하여 프로그래밍 방식으로 헤더 추가

View navHeaderView = navigationView.inflateHeaderView(R.layout.nav_header_main);
    headerUserName = (TextView) navHeaderView.findViewById(R.id.nav_header_username);
    headerMobileNo = (TextView) navHeaderView.findViewById(R.id.nav_header_mobile);
    headerMobileNo.setText("+918861899697");
    headerUserName.setText("Anirudh R Huilgol");

나를 위해 작동하지 않았으며 빈 nav_header가 표시됩니다. (텍스트, 이미지 없음)
막시무스

@maximus 나는 GitHub에 코드를 업로드하고 당신이 만든 실수를 찾을 수 있습니다.
Anirudh R. Huilgol.

4
  nav = ( NavigationView ) findViewById( R.id.navigation );

    if( nav != null ){
        LinearLayout mParent = ( LinearLayout ) nav.getHeaderView( 0 );

        if( mParent != null ){
            // Set your values to the image and text view by declaring and setting as you need to here.

            SharedPreferences prefs = getSharedPreferences("user_data", MODE_PRIVATE);
            String photoUrl = prefs.getString("photo_url", null);
            String user_name = prefs.getString("name", "User");

            if(photoUrl!=null) {
                Log.e("Photo Url: ", photoUrl);

                TextView userName = mParent.findViewById(R.id.user_name);
                userName.setText(user_name);

                ImageView user_imageView = mParent.findViewById(R.id.avatar);

                RequestOptions requestOptions = new RequestOptions();
                requestOptions.placeholder(R.drawable.ic_user_24dp);
                requestOptions.error(R.drawable.ic_user_24dp);

                Glide.with(this).load(photoUrl)
                        .apply(requestOptions).thumbnail(0.5f).into(user_imageView);

            }

        }
    }

도움이 되었기를 바랍니다.


4
val navigationView: NavigationView =  findViewById(R.id.nv)
val header: View = navigationView.getHeaderView(0)
val tv: TextView = header.findViewById(R.id.profilename)
tv.text = "Your_Text"

이것은 당신의 문제를 해결할 것입니다 <3


2

편집 : 23.0.1까지 디자인 라이브러리에서 작동하지만 23.1.0에서는 작동하지 않습니다.

기본 레이아웃 xml 에서 헤더보기를 설정 NavigationView하는 데 사용 app:headerLayout하도록 정의 할 것 입니다.

<android.support.design.widget.NavigationView
        android:id="@+id/navigation_view"
        android:layout_width="wrap_content"
        android:layout_height="match_parent"
        android:layout_gravity="start"
        app:headerLayout="@layout/nav_drawer_header"
        app:menu="@menu/navigation_drawer_menu" />

그리고는 @layout/nav_drawer_header이미지와 텍스트의 자리 표시자가 될 것입니다.

nav_drawer_header.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="170dp"
android:orientation="vertical">

<RelativeLayout
    android:id="@+id/headerRelativeLayout"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <ImageView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:scaleType="fitXY"
        android:src="@drawable/background" />

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="@dimen/action_bar_size"
        android:layout_alignParentBottom="true"
        android:layout_alignParentLeft="true"
        android:layout_alignParentStart="true"
        android:background="#40000000"
        android:gravity="center"
        android:orientation="horizontal"
        android:paddingBottom="5dp"
        android:paddingLeft="16dp"
        android:paddingRight="10dp"
        android:paddingTop="5dp">

        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            android:layout_marginLeft="35dp"
            android:orientation="vertical"
            android:weightSum="2">


            <TextView
                android:id="@+id/navHeaderTitle"
                android:layout_width="wrap_content"
                android:layout_height="0dp"
                android:layout_weight="1"
                android:textAppearance="?android:attr/textAppearanceMedium"
                android:textColor="@android:color/white" />

            <TextView
                android:id="@+id/navHeaderSubTitle"
                android:layout_width="wrap_content"
                android:layout_height="0dp"
                android:layout_weight="1"
                android:textAppearance="?android:attr/textAppearanceSmall"
                android:textColor="@android:color/white" />

        </LinearLayout>

    </LinearLayout>
</RelativeLayout>
</LinearLayout>

그리고 메인 클래스에서 일반적인 다른 뷰 ImageviewTextView마찬가지로 처리 할 수 ​​있습니다 .

TextView navHeaderTitle = (TextView) findViewById(R.id.navHeaderTitle);
navHeaderTitle.setText("Application Name");

TextView navHeaderSubTitle = (TextView) findViewById(R.id.navHeaderSubTitle);
navHeaderSubTitle.setText("Application Caption");

도움이 되었기를 바랍니다.


나는 그것을 구현했으며 아무런 문제없이 작동합니다. 나는 꽤 오랫동안 이런 식으로 사용하고 있습니다.
Kavin Prabhu 2015

나는 이것을 디자인 라이브러리 22.2.1에서 사용하고 잘 작동합니다. 절대로 23.1.0에 시도하지
Kavin 프라 부

시도했지만 응용 프로그램이 충돌합니다. navHeaderTitle.setText ( "Appliclation Name") 근처에 log cat 오류가 표시됨
kalai

지원 라이브러리 23.1.0을 사용하고 계십니까? 위의 주석에서 @PareshMayani가 제공하는 링크를 친절하게 살펴보십시오.
Kavin Prabhu 2015

2

나는 이것이 오래된 게시물이라는 것을 알고 있지만 이것이 누군가에게 도움이 될 것이라고 확신합니다.

다음을 수행하여 탐색보기의 headerView 요소를 간단히 가져올 수 있습니다.

 NavigationView mView = ( NavigationView ) findViewById( R.id.nav_view );

 if( mView != null ){
     LinearLayout mParent = ( LinearLayout ) mView.getHeaderView( 0 );

     if( mParent != null ){
        // Set your values to the image and text view by declaring and setting as you need to here. 
     }
 }

이것이 누군가에게 도움이되기를 바랍니다.


TextView를 얻는 방법? 그것은 것 mParent.findViewById()아니면 그냥 findViewById()?
Si8

mView.findViewById (R.id.text_view);
Ronnie

2

또한 Kotlinx 기능을 사용할 수 있습니다.

val hView = nav_view.getHeaderView(0)
hView.textViewName.text = "lorem ipsum"
hView.imageView.setImageResource(R.drawable.ic_menu_gallery)

2
   FirebaseAuth firebaseauth = FirebaseAuth.getInstance(); 

   NavigationView navigationView = (NavigationView) findViewById(R.id.nav_view);   //displays text of header of nav drawer.
    View headerview = navigationView.getHeaderView(0);

    TextView tt1 = (TextView) headerview.findViewById(R.id.textview_username);
    tt1.setText(firebaseauth.getCurrentUser().getDisplayName());//username of logged in user.  

   TextView tt = (TextView) headerview.findViewById(R.id.textView_emailid);
    tt.setText(firebaseauth.getCurrentUser().getEmail());    //email id of logged in user.

    final ImageView img1 = (ImageView) headerview.findViewById(R.id.imageView_userimage);
    Glide.with(getApplicationContext())
            .load(firebaseauth.getCurrentUser().getPhotoUrl()).asBitmap().atMost().error(R.drawable.ic_selfie_point_icon)   //asbitmap after load always.
            .into(new SimpleTarget<Bitmap>() {
                @Override
                public void onResourceReady(Bitmap resource, GlideAnimation<? super Bitmap> glideAnimation) {
                    img1.setImageBitmap(resource);
                }
            });

나는이 코드를 몇 가지 논리로 직접 만들었습니다 .100 % 작동합니다 ..... pls는 내 ans를 upvote합니다.

textview 및 imageview는 @ layout / nav_header_main.xml에서 가져온 것입니다.


1

그것은 오래된 게시물이지만 저에게는 새로운 것입니다. 그래서 간단합니다! 코드의이 부분에서 :

 public boolean onNavigationItemSelected(MenuItem item) {

}, 아래 나열된 예제의 ImageView를 포함하는 LinearLayout에 ImageView를 바인딩했습니다. 주의 : 새 프로젝트를 시작할 때 얻는 것과 동일한 코드이며 "Navigation Drawer Activity"템플릿을 선택합니다.

<ImageView
android:id="@+id/imageView"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:paddingTop="@dimen/nav_header_vertical_spacing"
android:src="@android:drawable/sym_def_app_icon" />

nav_header_main.xml 내부에 LinearLayout 및 ID를 제공했습니다 (제 경우에는 'navigation_header_container'를 선택 했으므로 다음과 같이 진행했습니다.

LinearLayout lV = (LinearLayout) findViewById(R.id.navigation_header_container);

    ivCloseDrawer = (ImageView) lV.findViewById(R.id.imageView);
    ivCloseDrawer.setOnClickListener(new View.OnClickListener()
    {
        @Override
        public void onClick(View v)
        {
            drawer.closeDrawer(GravityCompat.START);
        }
    });

참고 : onCreate (MainActivity) 전에 개인 ImageView ivCloseDrawer가 맨 위에 선언되어 있습니다.

잘 작동했습니다! 도움이되기를 바랍니다. 감사합니다.


1

바인딩을 사용하는 경우 할 수 있습니다.

val headerView = binding.navView.getHeaderView(0)
val headerBinding = NavDrawerHeaderBinding.bind(headerView)
headerBinding.textView.text = "Your text here"

0

NavigationView navigationView = (NavigationView) findViewById(R.id.nav_view); navigationView.addHeaderView(yourview);


1
안녕하세요 @ Роман Бяков, Stack Overflow에 오신 것을 환영합니다. 귀하의 기여에 감사드립니다. 질문이 4 개월 이상 지난 것을 알고 있습니까 :-)? 대답이 여전히 필요한지 확실하지 않습니다 ... 그러나 다른 대답에 대해서는 조금 더 설명하십시오. yourview귀하의 제안 은 무엇이며 OP는 귀하의 코드를 어디에 넣어야합니까? 그녀 가 줄 을 추가 해야하나요 아니면 같은 줄을 당신의 줄로 바꿔야 할까요? 이러한 설명은 답변을 시도하고 기여의 가치를 확인하는 데 도움이됩니다. 행운을 빕니다.
peter_the_oak

그만한 가치가있는 것은,이 문제에 대한 확실한 답은없고 여전히 활성화되어 있습니다.이 임의의 4 개월 기한이 지난 이후로 3 개의 새로운 게시물이 있다는 것을 알고 계십니까 :-)? 귀하의 의견이 필요한지 확실하지 않습니다 ... 그러나 향후 의견에 대해서는 기여를 완전히 낙담시키는 대신 기여를 개선하는 데 집중해야합니다.
SQLiteNoob

4 개월이 지난 많은 질문이 여전히 답을 얻습니다. 사실, 훨씬 더 오래된 (몇 년 된) 몇 가지 질문이 여전히 답을 얻고 있습니다! 그리고 그것은 좋은 일입니다. OP가 답을 얻었을지 모르지만 동일한 질문을 가지고 기존의 답변 풀에 대한 질문에 대한 답변이없는 사람들이 있습니다. StackOverflow는 동일한 질문의 재 게시를 권장하지 않으므로 새로운 답변은 동일한 질문을 가진 다른 사람들에게 도움이 될 것입니다. 또한 API는 시간이 지남에 따라 변경되며 오래된 답변에는 때때로 더 이상 사용되지 않는 코드가 포함됩니다.
ONE

0

다음은 헤더보기를 가져오고 데이터를 적절하게 설정하는 데 사용할 수있는 방법입니다.

val headerView: View? = navigationView.getHeaderView(0) // Index of the added headerView  

// Now you can access child views of the header view
val titleTextView: TextView? = headerView?.findViewById(R.id.titleTextView)
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.